[Python]Dijkstra 演算法實作

更新於 發佈於 閱讀時間約 10 分鐘

Dijkstra 演算法實作

Dijkstra 是用來找到從一個起點到其他所有點的最短路徑的演算法,適合靜態地圖。

實作步驟

  1. 定義地圖:使用圖(Graph)表示地圖。
  2. 初始化距離:起點距離設為 0,其餘點設為無限大。
  3. 遍歷節點:逐步找出距離最短的節點,更新相鄰節點的距離。
  4. 結束條件:所有節點被訪問完成。

程式碼

import heapq  # 引入 heapq 模組,實現最小堆優先隊列

def dijkstra(graph, start):
# 初始化最短距離字典,將所有節點的最短距離設為無窮大
distances = {node: float('inf') for node in graph}
distances[start] = 0 # 起點到自己的距離為 0

# 儲存已訪問的節點集合,避免重複處理節點
visited = set()

# 初始化優先隊列,將起點節點及其距離 0 加入隊列
priority_queue = [(0, start)] # 優先隊列中的元素是 (距離, 節點)

while priority_queue:
# 從優先隊列中取出距離最小的節點
current_distance, current_node = heapq.heappop(priority_queue)

# 如果當前節點已經訪問過,跳過該節點
if current_node in visited:
continue

# 將當前節點標記為已訪問
visited.add(current_node)

# 遍歷當前節點的所有鄰居
for neighbor, weight in graph[current_node].items():
# 計算從起點到鄰居的距離
distance = current_distance + weight
# 如果新計算出的距離比已知距離更短,則更新最短距離
if distance < distances[neighbor]:
distances[neighbor] = distance
# 將鄰居節點和其新的距離加入優先隊列
heapq.heappush(priority_queue, (distance, neighbor))

# 返回從起點到所有節點的最短距離
return distances

# 測試圖:節點間的連接與對應的邊的權重
graph = {
'A': {'B': 1, 'C': 4}, # 節點 A 與 B 的距離為 1,A 與 C 的距離為 4
'B': {'A': 1, 'C': 2, 'D': 5}, # 節點 B 與 A 的距離為 1,B 與 C 的距離為 2,B 與 D 的距離為 5
'C': {'A': 4, 'B': 2, 'D': 1}, # 節點 C 與 A 的距離為 4,C 與 B 的距離為 2,C 與 D 的距離為 1
'D': {'B': 5, 'C': 1} # 節點 D 與 B 的距離為 5,D 與 C 的距離為 1
}

start_node = 'A' # 設定起點為節點 A
# 呼叫 dijkstra 函數,計算從節點 A 出發的最短距離
shortest_distances = dijkstra(graph, start_node)

# 輸出結果,顯示從節點 A 到其他節點的最短距離
print("Shortest distances from node A:", shortest_distances)

測試輸出

對於這個測試例子,演算法會計算並返回從節點 A 到其他節點的最短距離,輸出結果應該如下:

Shortest distances from node A: {'A': 0, 'B': 1, 'C': 3, 'D': 4}

這表示:

  • 從 A 到 A 的距離是 0。
  • 從 A 到 B 的最短距離是 1。
  • 從 A 到 C 的最短距離是 3。
  • 從 A 到 D 的最短距離是 4。


程式碼詳細解說


1. 引入 heapq 模組

import heapq

heapq 是 Python 的標準庫,用於實現最小堆(min-heap)。在 Dijkstra 演算法中,我們使用最小堆來高效地選擇當前距離最小的節點。

2. 定義 dijkstra 函數

def dijkstra(graph, start):

這個函數接受兩個參數:

  • graph:一個字典表示圖,節點作為鍵,與該節點相連的鄰居節點及其邊的權重作為值。
  • start:起點節點,演算法從這個節點開始計算最短距離。

3. 初始化最短距離字典

distances = {node: float('inf') for node in graph}
distances[start] = 0 # 起點距離為 0
  • 我們使用 distances 字典來存儲每個節點到起點的最短距離。初始時,所有節點的距離設定為無窮大(float('inf')),表示尚未被訪問過。
  • 然後,將起點的距離設置為 0,因為起點到自身的距離是 0。

4. 初始化已訪問節點集合

visited = set()

visited 集合用來存儲已經訪問過的節點。這樣可以避免再次訪問同一節點。

5. 初始化優先隊列

priority_queue = [(0, start)]  # (距離, 節點)

priority_queue 是一個優先隊列,用來存儲尚未訪問的節點,並且根據距離進行排序(最小堆)。初始時,我們將起點節點和其距離(0)加入隊列。

6. 主迴圈

while priority_queue:
current_distance, current_node = heapq.heappop(priority_queue)

這是 Dijkstra 演算法的核心部分:

  • 當優先隊列不為空時,不斷循環。
  • 我們使用 heapq.heappop(priority_queue) 來取出當前距離最小的節點。這會返回一個二元組 (distance, node),即節點的距離和節點本身。

7. 如果節點已訪問過,跳過

if current_node in visited:
continue

如果當前節點已經在 visited 集合中,說明它已經被處理過,這時就跳過它,繼續處理下一個節點。

8. 標記節點為已訪問

visited.add(current_node)

將當前節點標記為已訪問,這樣就不會再次處理它。

9. 更新鄰居節點的距離

for neighbor, weight in graph[current_node].items():
distance = current_distance + weight
if distance < distances[neighbor]:
distances[neighbor] = distance
heapq.heappush(priority_queue, (distance, neighbor))

對於當前節點的每一個鄰居:

  • graph[current_node].items() 返回當前節點的所有鄰居及其邊的權重。
  • 計算從起點到鄰居的距離:distance = current_distance + weight
  • 如果計算出的距離比鄰居當前的距離短,則更新鄰居的最短距離,並將這個鄰居節點及其新的距離加入優先隊列。

10. 返回最短距離字典

return distances

當所有節點都處理完後,返回 distances 字典,其中包含了從起點到每個節點的最短距離。

11. 測試地圖與呼叫函數

# 節點間的連接與距離
graph = {
'A': {'B': 1, 'C': 4},
'B': {'A': 1, 'C': 2, 'D': 5},
'C': {'A': 4, 'B': 2, 'D': 1},
'D': {'B': 5, 'C': 1}
}

start_node = 'A'
shortest_distances = dijkstra(graph, start_node)
print("Shortest distances from node A:", shortest_distances)

這段程式碼定義了一個圖 graph,其中每個節點的鄰居和對應的邊權重以字典的形式給出。然後,呼叫 dijkstra 函數來計算從節點 A 出發的最短距離,並將結果打印出來。


avatar-img
128會員
215內容數
本業是影像辨識軟體開發,閒暇時間進修AI相關內容,將學習到的內容寫成文章分享。
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
螃蟹_crab的沙龍 的其他內容
A* (A-Star)演算法實作 A* (A-Star)是一種啟發式搜尋演算法,除了最短距離,還會考慮目標的估計距離(啟發式函數)。 實作步驟 定義地圖和啟發式函數。 初始化開放節點(Open Set)和已訪問節點(Closed Set)。 計算代價函數 f(n) = g(n) + h(n
A* (A-Star)演算法實作 A* (A-Star)是一種啟發式搜尋演算法,除了最短距離,還會考慮目標的估計距離(啟發式函數)。 實作步驟 定義地圖和啟發式函數。 初始化開放節點(Open Set)和已訪問節點(Closed Set)。 計算代價函數 f(n) = g(n) + h(n
你可能也想看
Google News 追蹤
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
題目敘述 Find K-th Smallest Pair Distance 給定一個輸入陣列nums和 參數k。 請找出第k小的pair distance是多少? pair distance定義為 abs( nums[i] - nums[j]), i 不等於j 也就是任意兩陣列元素差值的絕對值
Thumbnail
題目敘述 All Ancestors of a Node in a Directed Acyclic Graph 給定一個有向無環圖,請找出每個點的祖先,以陣列的形式返回答案。
Thumbnail
題目敘述 Triangle 題目會給我們一個三角形的二維陣列triangle ,每個元素分別代表每個格子的成本,請問我們從最頂端到底部的下墜路徑的最小成本總和是多少? 每次下墜到下一排的時候,可以有兩種選擇: 1.往左下方的格子點移動。 2.往右下方的格子點移動。 測試範例 Examp
Thumbnail
題目敘述 Binary Tree Maximum Path Sum 給定一個二元樹,請找出最大的區間路徑和是多少? 註: 區間路徑和 = 某個節點a -> 某個節點b的路徑節點值總和。
Thumbnail
Minimum Path Sum 給定一個矩陣,每個格子點代表經過的對應成本。 每回合可以往右移動一格或往下移動一格。 請問從起點左上角 走到 終點右下角的最小路徑成本總和是多少?
Thumbnail
題目敘述 題目會給我們一顆二元樹的根節點。請問在這棵樹中,之字型走法的路徑長度最大值是多少? 如果無解,請返回 零。 註: 之字型走法就是有一段路徑,都是由連續的 左右左右...,或者 右左右左...所構成的路徑。(看下方的測試範例會更清楚題目的定義) 題目的原文敘述 測試範例 E
Thumbnail
題目敘述 題目會給定一顆二元樹的根結點Root node,和指定的目標值targetSum。 問我們能不能從二元樹裡面找到一條從根結點到葉子結點的路徑,其路徑上的節點值總和恰好為targetSum? 可以的話,返回True。 無解的話,返回False。 題目的原文敘述 測試範例 E
Thumbnail
題目敘述 題目會給定一個指定高度和寬的方格版,還有一顆小球的起始位置,和最大移動步數。 小球每一步可以選擇向上、下、左、右移動一格,請問小球能走到方格版界外的路徑方法數總共有幾種? 方法數可能很大,題目要求,最後回傳答案時,先對10^9+7做除法取餘數再回傳。 題目的原文敘述 約束條件
Thumbnail
題目敘述 題目的情境是設計並且實現一個包含所有正整數的數據流,以set集合的方式存在。 數據流 = {1, 2, 3, 4, ..., ∞} 要求我們去實現定義好的function介面: SmallestInfiniteSet()建構子,初始化這個包含所有正整數的數據流。 int po
Thumbnail
這題也是滿經典的DP動態規劃教學案例和題目,就順便複習一下吧。 題目敘述 題目會給我們兩個字串text1, text2。 要求我們找出兩個字串的最長共同子序列,並且返回最長共同子序列的長度。 如果彼此沒有共同子序列,則返回0。 題目的原文敘述 測試範例 Example 1: In
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
題目敘述 Find K-th Smallest Pair Distance 給定一個輸入陣列nums和 參數k。 請找出第k小的pair distance是多少? pair distance定義為 abs( nums[i] - nums[j]), i 不等於j 也就是任意兩陣列元素差值的絕對值
Thumbnail
題目敘述 All Ancestors of a Node in a Directed Acyclic Graph 給定一個有向無環圖,請找出每個點的祖先,以陣列的形式返回答案。
Thumbnail
題目敘述 Triangle 題目會給我們一個三角形的二維陣列triangle ,每個元素分別代表每個格子的成本,請問我們從最頂端到底部的下墜路徑的最小成本總和是多少? 每次下墜到下一排的時候,可以有兩種選擇: 1.往左下方的格子點移動。 2.往右下方的格子點移動。 測試範例 Examp
Thumbnail
題目敘述 Binary Tree Maximum Path Sum 給定一個二元樹,請找出最大的區間路徑和是多少? 註: 區間路徑和 = 某個節點a -> 某個節點b的路徑節點值總和。
Thumbnail
Minimum Path Sum 給定一個矩陣,每個格子點代表經過的對應成本。 每回合可以往右移動一格或往下移動一格。 請問從起點左上角 走到 終點右下角的最小路徑成本總和是多少?
Thumbnail
題目敘述 題目會給我們一顆二元樹的根節點。請問在這棵樹中,之字型走法的路徑長度最大值是多少? 如果無解,請返回 零。 註: 之字型走法就是有一段路徑,都是由連續的 左右左右...,或者 右左右左...所構成的路徑。(看下方的測試範例會更清楚題目的定義) 題目的原文敘述 測試範例 E
Thumbnail
題目敘述 題目會給定一顆二元樹的根結點Root node,和指定的目標值targetSum。 問我們能不能從二元樹裡面找到一條從根結點到葉子結點的路徑,其路徑上的節點值總和恰好為targetSum? 可以的話,返回True。 無解的話,返回False。 題目的原文敘述 測試範例 E
Thumbnail
題目敘述 題目會給定一個指定高度和寬的方格版,還有一顆小球的起始位置,和最大移動步數。 小球每一步可以選擇向上、下、左、右移動一格,請問小球能走到方格版界外的路徑方法數總共有幾種? 方法數可能很大,題目要求,最後回傳答案時,先對10^9+7做除法取餘數再回傳。 題目的原文敘述 約束條件
Thumbnail
題目敘述 題目的情境是設計並且實現一個包含所有正整數的數據流,以set集合的方式存在。 數據流 = {1, 2, 3, 4, ..., ∞} 要求我們去實現定義好的function介面: SmallestInfiniteSet()建構子,初始化這個包含所有正整數的數據流。 int po
Thumbnail
這題也是滿經典的DP動態規劃教學案例和題目,就順便複習一下吧。 題目敘述 題目會給我們兩個字串text1, text2。 要求我們找出兩個字串的最長共同子序列,並且返回最長共同子序列的長度。 如果彼此沒有共同子序列,則返回0。 題目的原文敘述 測試範例 Example 1: In