【图算法】Dijkstra —— 最短路径

Dijkstra算法是一种单源最短路径算法,适用于非负权重的边。算法以起始点为中心逐步扩展,找到最短路径。核心思路是使用优先队列(小顶堆)存储待更新的节点,结合距离数组dist和访问标记visited更新最短路径。当优先队列为空时,所有节点的最短路径已确定。本文还提供了Python和C++的参考代码。
摘要由CSDN通过智能技术生成

Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径。主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止。算法要求所有边的权重必须是非负的。

问题:
给定三个集合V、 E、 W,其中,V存放着所有顶点的index/name(下面代码里的V为节点数),E存放所有边的信息,W存放着E中对应边的权重(下面代码里把E和W合并到在一起了)。现在给出V中的一个顶点A,求从A出发到其他顶点的最短路径,即经过边的权重和最小。

基本思路:

  1. 建立优先队列pq(小顶堆),队列里存储在当前条件下(表示以后可能还会被更新)离源点最近距离的节点及其距离(距离是为了供优先队列比较大小用),初始只有源点,其距离值为0;还需要一个dist数组以及visited数组,数组长度均为节点个数;dist用来记录当前状态下各个节点离源点的最短路径,初始除源点外的值为无穷大,源点的值为0;visited用来记录各个节点是否已经被优先队列弹出过,若为True则表示当前节点被弹出过。
  2. 如果优先队列不为空,则弹出优先队列队顶元素,即下一个已经更新过最短路径的节点,如果该节点已经被访问了,即表示在弹出该节点之前该节点有一个比当前最短路径更近的情况被弹出过,则此情况应该丢弃。这也说明我们每次加入优先队列的元素不一定就是最短路径的情况,但是优先队列优先弹出最短路径,这也是为什么我们用优先队列而不是普通队列的原因。
  3. 如果弹出的节点之前没有被访问过/弹出过,那么该节点的最短路径就被确定了,然后遍历与该节点直接相连的节点,并更新这些节点的最短路径。
  4. 如果优先队列为空,则说明所有节点的最短路径都已经被确定了,结束循环。

原始思路1:

  1. 遍历V - S中的所有点,求出从初始点A只经过S中顶点到达该点的路径权重和dist,若不存在路径,则dist设置为 ∞ \infty
  2. 找出第一步中的最小的dist,将该dist对应的顶点放入S中,该dist值放入U中;
  3. 重复1、2直到V等于S为止。

原始思路2:

  1. 遍历S中的点,找出所有E中从该点出发但是终点不在S中的边,算出对应的dist;
  2. 找出第一步中的最小dist,将该dist对应的顶点放入S中,该dist值放入U中;
  3. 重复1、2直到V等于S为止。

Python 参考源码:

import sys
import heapq
MAX_INT = sys.maxsize

def add_edges(adj, u, v, w):
    """build graph by adding edges"""
    adj[u].append((v, w));
    adj[v].append((u, w));

def dijkstra(adj, V, src):
    # 用优先队列(默认小顶堆)完成最小dist节点的动态选取
    pq = []
    # dist表示src到每个顶点的最短路径,初始为MAX_INT
    dist = [MAX_INT]*V
    # visited记录节点是否被当作最小dist节点访问过
    visited = [False]*V

    # 优先队列初始只有起始点,起始点的dist为0
    heapq.heappush(pq, (0, src))
    dist[src] = 0

    while pq:
        # 弹出当前距离源点dist最小而且未被访问的节点
        wt, u = heapq.heappop(pq)
        if visited[u]:
            continue
        visited[u] = True
        # 遍历该节点直接相连的节点并更新对应节点的dist
        for v, w in adj[u]
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值