无向图求最短路径——Dijkstra(迪杰斯特拉算法)

    Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个顶点到其他所有顶点的最短路径。

   问题描述: 如下无向图, 若从顶点1开始计算到其余各顶点的最短路径

   

 

 首先需要3个辅助数组:  

  • dist[] : 记录从顶点1开始到其余各顶点的最短路径
  • visited[] : 记录该顶点是否被访问过, 初始值设为0
  • path[] : 记录该顶点最短路径的前驱顶点

  求最短路径步骤:

    ①初始化数组:计算从顶点0到其余各顶点的路径长度
 

顶点1234567
dist[]0121614
path[]-11-1-1-111
visited[]1000000

 

   ②由于dist数组中目前最小值为顶点2,计算从顶点2到其余各顶点的距离, 若小于当前dist数组中的值, 则更新dist和path数组;若大于当前dist数组的值, 则不做改变

顶点1234567
dist[]012221614
path[]-112-1-111
visited[]1100000

     ③选择dist数组中未被访问过的最小值顶点7, 同上

顶点1234567
dist[]01222221614
path[]-112-1711
visited[]1100001

 

   ④选择dist数组中未被访问过的最小值顶点6

顶点1234567
dist[]01222181614
path[]-112-1611
visited[]1100011

 

   ⑤选择dist数组中未被访问过的最小值顶点5

顶点1234567
dist[]0122222181614
path[]-1125611
visited[]1100111

 

   ⑥选择dist数组中未被访问过的最小值顶点3或者4都可以,  如选择3

顶点1234567
dist[]0122222181614
path[]-1125611
visited[]1110111


   ⑦选择dist数组最后一个顶点4

顶点1234567
dist[]0122222181614
path[]-1125611
visited[]1111111

 

到这里为止, visited数组元素全为1, 即所有顶点都被访问过, 最短路径见如上图。

迪杰斯特拉算法(Dijkstra)是一种贪心算法,用于解决最短路径问题。它可以处理有权有向图或无向图,但不允许有负权边(权重必须为非负数)。 算法思路: 1. 从起点开始,初始化所有节点的距离为无穷大,起点距离为0; 2. 将起点加入“已访问”集合; 3. 对于起点的所有邻居节点,更新它们的距离(如果通过当前节点到达邻居节点的距离小于邻居节点原有的距离,则更新邻居节点的距离); 4. 从未访问集合中选择距离起点最近的节点,加入“已访问”集合; 5. 重复步骤3和4,直到所有节点都被加入“已访问”集合或者没有与起点相连的节点。 算法实现: Dijkstra算法的实现通常使用优先队列(PriorityQueue)来维护未访问集合中距离起点最近的节点。具体实现步骤如下: 1. 创建一个空的优先队列Q,将起点加入Q中,并设置起点到自身的距离为0; 2. 创建一个数组dist[],用于保存起点到各个节点的距离,初始化为无穷大; 3. 创建一个数组visited[],用于标记节点是否被访问过,初始化为false; 4. 将dist[起点]的值设置为0; 5. 当Q不为空时,重复以下步骤: a. 从Q中取出距离起点最近的节点u; b. 如果节点u已经被访问过,则跳过此次循环; c. 将节点u标记为已访问; d. 对于节点u的每个邻居节点v,如果节点v未被访问过并且通过节点u到达节点v的距离小于dist[v],则更新dist[v]的值; e. 将节点v加入Q中。 6. 最终,dist数组中保存的就是起点到各个节点的最短距离。 Dijkstra算法的时间复杂度为O(ElogV),其中E为边数,V为节点数。这是因为算法需要对每个节点的所有邻居节点进行遍历,而优先队列的插入和删除操作的时间复杂度为O(logV)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值