ダイクストラ法(だいくすとらほう、英: Dijkstra's algorithm)はグラフ理論における辺の重みが非負数の場合の単一始点最短経路問題を解くための最良優先探索によるアルゴリズムである。辺の重みに負数を含む場合はベルマン-フォード法などが使える。辺の重みが全て同一の非負数の場合は幅優先探索が速く、線形時間で最短路を計算可能である。また、無向グラフで辺の重みが正整数の場合は、Thorupのアルゴリズム[1]によって線形時間での計算が可能であるが、実用性はあまり高くない[要出典]。
ダイクストラ法はグラフ上の2頂点間の最短経路を求めるアルゴリズムで、1959年エドガー・ダイクストラによって考案された。
応用範囲は広くOSPFなどのインターネットルーティングプロトコルや、カーナビの経路探索や鉄道の経路案内においても利用されている。
なお最短経路長の推定値を事前に知っているときは、ダイクストラ法の改良版であるA*アルゴリズムを用いて、より効率的に最短経路を求めることができる。
擬似コード[編集]
ダイクストラ法の擬似コードは以下の通りである。
は頂点の集合(もしくは優先度付きキュー)。
,
は頂点。
はスタートとなる頂点からの最短経路の長さ。
は最短経路をたどる際の前の頂点。
グラフ
、スタートとなる頂点
、および
から
への辺の長さ
を入力として受け取る
// 初期化
各頂点
に対し、
ならば
、それ以外は
各頂点
に対し、
「無し」
に
の頂点を全て追加
// 本計算
while (
が空ではない )
から
が最小である頂点を取り除く
for each (
からの辺がある各頂点
)
if (
)
「
から
が最小である頂点を取り除く」の部分は、オリジナルは単純に集合内を探索するアルゴリズムだが、
を優先度付きキュー(フィボナッチヒープ)にすることで、より計算量が減る。ただしその場合、
に関する部分を少し変更する必要がある。下記は優先度付きキューを用いたことを想定し上記を書き換えたダイクストラ法の擬似コードである。
// 初期化
for (
)
ならば
、それ以外は
「無し」
// 本計算
while (
が空集合ではない )
から
が最小である頂点
を取り出す
for each (
からの辺がある各
)
if (
)
「for each (
からの辺がある各頂点
)」の部分は「for each (
からの辺がある各頂点
)」でも同じ結果が得られるが、
ではあるものの、逆に計算量が増えてしまう場合もあり得ることに注意。
GCC の C++ の priority_queue はペアリングヒープを使用しているため[3]、これを使うとフィボナッチヒープと同等の計算量のコードが簡単に実装できる。
計算量[編集]
計算量は以下の通り。
- オリジナル :

- 優先度付きキュー(二分ヒープ):

- 優先度付きキュー(フィボナッチヒープ):

優先度付きキューを使用した場合の計算量としては、以下の合算である。下記説明は上記擬似コードに基づき、ループ回数も考慮に入れている。
- 「
」:二分ヒープもフィボナッチヒープも
。ただし、二分ヒープは追加を V 回繰り返す実装にすると
。
- 「
が最小である頂点
」:二分ヒープもフィボナッチヒープも 
- 「
から
を取り除く」:二分ヒープもフィボナッチヒープも 
- 「
」:二分ヒープは
、フィボナッチヒープは 
アルゴリズムの解説[編集]
話を簡単にするため、
の各頂点
に対し、
と
を結ぶ辺は最大一つしかないとする。
と
を結ぶ辺があれば、その辺を
と書く。
最短経路問題は、ビー玉と紐を用いて解くことができる。
まずビー玉を頂点、紐を辺にするグラフを工作する。
グラフを板の上に置き、スタートの頂点にあたるビー玉だけをつまむ。
グラフが置かれている板を取り除くと、グラフは自由落下を始めるが、
スタートにあたるビー玉を持っているので、スタート地点から近いビー玉から順に落下が止まる。
ゴールにあたるビー玉が止まったとき、ゴールにあたるビー玉はスタートにあたるビー玉まで紐で一直線で結ばれている。
この直線が最短経路である。
落下が止まった頂点
に対し、vを支えている頂点
が存在する。
を
の「直前の頂点」と呼ぶことにする。
以下簡単のため、直前の頂点は1つしかないと仮定して話を進めるが、
一般の場合も同様である。
ダイクストラのアルゴリズムは、上述の解法をコンピュータ上でシミュレートしたものである。
グラフ
、スタートとなる頂点
、および各辺
の長さ
が入力として与えられると、
このアルゴリズムは上述の解法をシミュレートし、「落下」が停止した頂点から順に、その頂点の直前の頂点が何であるかを出力する。
ゴールとなる頂点の「落下」が停止したところで、ダイクストラのアルゴリズムを止めれば、
あとはゴールから順に、直前の頂点、さらにその直前の頂点とたどっていくことで、
ゴールとスタートをつなぐ最短経路(の一つ)を求めることができる。
以上の考察から分かるように、ダイクストラ法では「次に落下が停止する頂点」(とその直前の頂点)を求めることが重要である。
そこでこれらを効率的に求める方法を説明する。
現時点で「落下」が停止している頂点の集合をHとし、
各頂点
に対して
内での
と
との最短距離を
とする(
内で
から
に到達できないときは
とする)。
さらに、
に隣接している頂点の集合を
とする。
ここで頂点
が
に隣接しているとは、
と
内のいずれか頂点を結ぶ辺が存在することを指す。
次に落下が停止する頂点を
とし、
の直前の頂点を
とすると、
以下が成立することが分かる。
は
に属し、しかも
内で
が最小になる頂点である。
から
への最短経路は
を通る。
そこでダイクストラ法では、「次に落下が停止する頂点」(とその直前の頂点)を効率的に求めるために、
以下の3種類のデータを管理する。
- 集合
と集合 
- 各頂点
に対して
内での
から
への最短距離
。
- 各
に対し、
の直前の頂点 
ダイクストラ法では、
内で
が最小になる頂点
(
次に落下が停止する頂点)を求めて
を出力し、それにあわせて管理しているデータを更新し、そしてさらに次に落下が停止する頂点を求める、という操作を繰り返す。
そこで最後に、ダイクストラ法が管理しているデータを更新する方法を述べる。
内で
が最小になる頂点
(
次に落下する頂点)が求まったら、まず
を
に追加する。それにあわせて
から
を除去し、
に隣接していてかつ
に属さない頂点を
に加える。
を
に追加した結果「
内での
から
への最短距離」である
が変化するのは、
と
とを結ぶ辺がある場合に限られる。
を
に追加後の「
内での
から
への最短距離」は次のいずれかと一致する。
- (a)
を
に追加する前の段階での
から
への最短経路
- (b)
を
に追加する前の段階での
から
への最短経路を通った後で
と
を結ぶ辺を通るという経路
(a)の長さは
を
に追加する前の段階での
に一致し、
一方(b)の長さは
を
に追加する前の段階での
に
を加えた長さである。
以上の考察より、
および
を次のように更新すればよいことがわかる:
から
への辺が存在する各
に対し、
なら、
を「
」に更新し、さらに
を
に更新。
- そうでなければ何もしない。
参考文献[編集]
関連項目[編集]