数学建模法-Dijkstra算法

一、引言

哈喽大家好,今天要讲的是图论中的一个经典的算法。是一种叫Dijkstra算法的东东。这个算法是干什么用的呢。首先大家先看下面这幅图:

这个东西是什么呢。我们可以这样理解,假如A到F表示6个地点。那些连接线就是道路。连接线上的数字就是两个地点间的距离。这样讲是不是很直观呢。好了,假如博主家在A点,博主的女神家在F点,有一天博主想去女神家,就有很多条路线可以走。可是博主很懒诶,肯定就想走最短的路线。那么,怎么才能很快的就求解出最短的路线呢。Dijkstra算法就是用来解决这样的问题的。

二、Dijkstra算法的思想

Dijkstra算法的思想其实很直观,就是我从A点出发,发现可以走的路就只有C和B了,那么我肯定就要走最近的那条路,也就是C(同时记录C与A的距离)。接下来,我们从C点出发,可以走的路有B和D和E,再选择出最近的一条路,也就是B点(同时也记录C与B的距离)。通过每次不断的走最短的路线。最后走到F的路线也肯定是最短的。这就是Dijkstra算法的思想。当然讲起来很简单,计算起来有时候也会遇到一些其他因素。接下来我会尽可能通俗易懂的讲这个算法的过程讲清楚。

三、Dijkstra算法步骤

接下来的讲解, 可能有点生涩,我会先写出来再慢慢解释。

首先,我们要先确定我们的起点。在这里我们定我们的起点为A。但这样还不够,我们要写成(端点,端点与起点之间的距离)的形式。因此,我们的A就要写为(A, 0)。

准备工作做完了,我们开始吧。

(1) 首先从(A, 0)开始,A端点所连接的两个点分别是(B, 5)和(C, 1)。我们先把(B, 5)和(C, 1)存到Box里,并且根据大小来排序(最小排左边)。如下图

显然,(C, 1)是最小的,因此,A点到下一个点的最短路线点是(C, 1)。我们不妨再创建一个Box吧,称之为Box3吧。称Box3是因为有3行。第一行表示图上所有的地点,第二行表示对于第一行地点的前一个最短路线点,第三行表示第一行的地点与原点的距离。如下:

大家可以看到, 第二行第一列,因为我们的A就是起点,在A之前并没有什么点,因此打×,下面的0表示无的意思。因为我们现在求出我们的(C, 1)是我们的下一个最短路线点,并且C是从A走过去的,故第二行第三列是A ,其下面的1表示从起点到C的最短距离为1。

(2) 好了,现在,既然确定了(A, 0)的下一个点是(C, 1)。接下来,我们要从(C, 1)出发,我们看看,跟(C, 1)连接的点有(B, 3)点(注意,是3不是2,我们说过是与起点的距离,即A-C-B)和(D, 5)点以及(E, 9)。把其填入Box里面并排序。如下:

咦,这时候你可能会有疑问,为什么(C, 1)不见了,而且有两个B,一个是(B, 3)一个是(B, 5)。别急,我正要讲。当我们确定好最短距离的点后,我们就将其从Box里面剔除掉,因为C已经被用过了。而至于为何有两个B。大家还记得,当我们直接从A到B的时候,距离是5;而当我们从A到C到B的时候,距离是3。因此就有两个了。现在,在这个Box里面我们找到最小的数,就是(B, 3)。这样我们就可以在Box3也填入我们的数,如下:

(3) 好的,让我们继续,接下来我们从(B, 3)开始,由于B的连接点只剩(D, 4)(已经用过的点不能再用,故A和C不算。毕竟我们肯定不走回头路呀hhh)。填入Box并排序,可以得到(D, 4)是最小的数,填入Box3中。如下

         (4) 好了,接下来我们从(D, 4)出发,跟(D, 4)连接的有(E, 7)和(F, 10),填入Box并排序,可以得到(E, 7)是最小的数,填入Box3中。如下:

(5) 好,接下来我们从(E, 7)出发,诶这时候发现,E已经没有可连接的点了。那Box里面只剩(F, 10)了。把Box3最后一列填完。那我们就找到了最短去博主女神家的路了。

现在,我们知道从A到F的最短路线和距离分别是A-C-B-D-F和10。现在我们要谈谈Box3,当我们完成这个表后,我们不仅可以马上知道博主家到博主女神家的最短路线。还能知道任意点到起点的最短路线呢。比如说,我们想知从A到E的最短路线。看Box3,E的上一点是D,D的上一点是B,B的上一点是C,C的上一点是A。这样就得到A-C-B-D-E是最短路线。而Box3中E最下方的7就表示最短距离值哦。现在你们是不是搞懂了呢~^_^。

四、Dijkstra的Matlab实现

function [distance, path] = dijkstra(A, sn, en)
% [DISTANCE,PATH] = DIJKSTRA(A, SN, EN)
% returns the distance and path between the start node and the end node.
%
% A: adjcent matrix
% sn: start node
% en: end node

%% 初始化
%节点的数量n
n = length(A);
%以sn为起点的矩阵(distance vector)
D = A(sn,:);

vi = ones(1, n);    %让节点都可见
vi(sn) = 0;         %起点节点是不可见的 

%parent:即Box3中的第二行
parent = zeros(1, n);

%% 计算最短距离

for i = 1: n-1
    temp = zeros(1, n);
    count = 0;
    %把distance vector里面非顶点的距离值放进temp,以便后续比大小取出最短距离
    for j = 1: n
        if(vi(j))
            temp = [temp(1: count) D(j)];
        else
            temp = [temp(1: count) inf];
        end
        count = count + 1;
    end

    %找出最短距离的点,并设定为下次路径的顶点
    [~, index] = min(temp);
    vi(index) = 0;%让顶点不可见
    for k = 1: n
        if A(index, k) + D(index) < D(k)
            D(k) = A(index, k) + D(index);
            parent(k) = index;  %算出k的上一层最短路径点,即Index
        end
    end
end
%%迭代完成后,distance矩阵的数值都是相对应的最短距离
distance = D(en);

%%求出最短路径

path = [];
t = en; path(1) = t; count = 1;
while t ~= sn && t > 0
    p = parent(t);
    path = [p path(1:count)];    %从path里面 不断往左边放路径点,最右边是终点
    t = p;
    count = count + 1;
end
path(1) = sn;
path = path(1: count);

验证一下,我们根据这篇的图例写出邻接矩阵(这里我当大家知道什么是邻接矩阵,不知道的自己百度一下哈)。得到如下:

A = [0 5 1 5 9 inf;
     5 0 2 1 inf inf;
   1 2 0 4 8 inf;
   inf 1 4 0 3 6;
   inf inf 8 3 0 inf;
   inf inf inf 6 inf 0];

然后,因为我们要算A到F的最短距离,因此sn = 1, en = 6。故在matlab中输入:

[distance, path]=dijkstra(A, 1, 6)

  最后就得到如下结果:

distance =
    10
path =
     1     3     2     4     6

1-3-2-4-6翻译过来也就是A-C-B-D-F。跟我们前面分析的一致。故正确。

原文地址:https://www.cnblogs.com/Qling/p/9310263.html

时间: 2024-08-12 10:24:57

数学建模法-Dijkstra算法的相关文章

数学建模方法-Floyd算法

一.引言 哈喽大家好,今天要给大家讲的是Floyd算法.在那之前,大家还记得我们上一章讲的内容吗,就是那个Dijkstra算法,用来解决从A点到B点的最短路径问题.我们还给出了Matlab代码.Floyd算法也是用来处理最短路径问题的.它的理念跟Dijkstra有点不一样,但是最终的结果是一样的.Floyd算法主要是用到了动态规划的思想.在这里博主不打算讲到很抽象很高深的东西(毕竟博主也不是专业的),仅仅通过比较通俗易懂的方式来给大家讲解这个算法的思想(如果有问题大家帮忙指出来哈).本文的图片和

数学建模算法概括

目录 数学模型按数学方法分类 数学建模十大算法 建模思想 预测与预报 评价与决策 分类与判别 关联与因果 优化与控制 数学模型按数学方法分类 几何模型(球面积分,曲面积分) 分形理论(常用) 图论模型(优化类,规划类,决策类问题) 有一类线性规划类问题可用图论模型解决,最短路径 → 时间最短 or 路径最短 微分方程模型(预测人口增长,传热导热问题) 概率问题(彩票) 最优控制模型(药物疗效) 规划论模型(投资问题) 马氏链模型(概率模型) 前后不关联的概率模型 数学建模十大算法 蒙特卡罗算法

数学建模算法理论+程序

数学建模的各类算法汇总,带书签!文字可复制. 01 线性规划 02 整数规划 03 非线性规划 04 动态规划 05 图与网络 06 排队论 07 对策论 08 层次分析法 09 插值与拟合 10 数据的统计描述和分析 11 方差分析 12 回归分析 13 微分方程建模 14 稳定状态模型 15 常微分方程的解法 16 差分方程模型 17 马氏链模型 18 变分法模型 19 神经网络模型 20 偏微分方程模型 21 目标规划 22 模糊数学模型 23 现代优化算法 24 时间序列模型 25 贮存

余胜威《MATLAB数学建模经典案例实战》2015年版

内容介绍 本书全面.系统地讲解了数学建模的知识.书中结合历年全国大学生数学建模竞赛试题,采用案例与算法程序相结合的方法,循序渐进,逐步引导读者深入挖掘实际问题背后的数学问题及求解方法.在本书案例的分析计算中巧妙地结合了MATLAB等工具,并采用不同的算法进行模型求解,达到异曲同工之妙.本书结合实际,对网上讨论的很多疑难问题也做了解答. 本书共25章,分3篇.主要内容有:MATLAB基础知识.LINGO基础知识.SPSS基础知识.数学建模基础理论及算法设计.基于LINGO的基础理论及算法设计.企业

数学建模常用的十大算法

数学建模常用的十大算法==转 (2017-07-16 11:26:14) 转载▼ 1. 蒙特卡罗算法.该算法又称随机性模拟算法,是通过计算机仿真来解决问题的算法,同时可以通过模拟来检验自己模型的正确性,几乎是比赛时必用的方法. 2. 数据拟合.参数估计.插值等数据处理算法.比赛中通常会遇到大量的数据需要处理,而处理数据的关键就在于这些算法,通常使用MATLAB 作为工具. 3. 线性规划.整数规划.多元规划.二次规划等规划类算法.建模竞赛大多数问题属于最优化问题,很多时候这些问题可以用数学规划算

数学建模学习笔记(建模中的十大常用算法总结)

数学建模中的十大常用算法 1.    蒙特卡洛方法: 又称计算机随机性模拟方法,也称统计实验方法.可以通过模拟来检验自己模型的正确性. 2.    数据拟合.参数估计.插值等数据处理 比赛中常遇到大量的数据需要处理,而处理的数据的关键就在于这些方法,通常使用matlab辅助,与图形结合时还可处理很多有关拟合的问题. 3.    规划类问题算法: 包括线性规划.整数规划.多元规划.二次规划等:竞赛中又很多问题都和规划有关,可以说不少的模型都可以归结为一组不等式作为约束条件,几个函数表达式作为目标函

数学建模系列:模拟退火算法

引言 模拟退火算法基于Monte-Carlo迭代求解策略的一种随机寻优算法,其出发点是基于物理中固体物质的退火过程与一般组合优化问题之间的相似性.模拟退火算法从某一较高初温出发,伴随温度参数的不断下降,结合概率突跳特性在解空间中随机寻找目标函数的全局最优解,即在局部最优解能概率性地跳出并最终趋于全局最优.模拟退火算法是一种通用的优化算法,理论上算法具有概率的全局优化性能,目前已在工程中得到了广泛应用,诸如VLSI.生产调度.控制工程.机器学习.神经网络.信号处理等领域. 模拟退火算法是通过赋予搜

python 版 mldivide matlab 反除(左除)《数学建模算法与程序》Python笔记

今天在阅读数学建模的时候看到了差分那章 其中有一个用matlab求线性的代码,这里我贴出来 这里我送上 Python代码 In [39]: import numpy as np ...: from scipy.optimize import nnls ...: x = np.array([[1,2,3,4,5],[1,1,1,1,1]]) ...: x = x.T ...: y = np.array([11,12,13,15,16]) ...: nnls(x,y) ...: Out[39]: (

数学:Dijkstra算法

一.最短路径的最优子结构性质 该性质描述为:如果P(i,j)={Vi....Vk..Vs...Vj}是从顶点i到j的最短路径,k和s是这条路径上的一个中间顶点,那么P(k,s)必定是从k到s的最短路径.下面证明该性质的正确性. 假设P(i,j)={Vi....Vk..Vs...Vj}是从顶点i到j的最短路径,则有P(i,j)=P(i,k)+P(k,s)+P(s,j).而P(k,s)不是从k到s的最短距离,那么必定存在另一条从k到s的最短路径P'(k,s),那么P'(i,j)=P(i,k)+P'(