출처: 픽사베이
🚀최단 경로 탐색
지난 시간에는 그래프와 최소 신장 트리(MST)를 구하는 알고리즘인 프림, 크루스칼 알고리즘을 알아보았다. (그래프 알고리즘이란 + 최소 신장 트리(MST))
이번 포스팅은 그래프 최단 거리를 구하는 알고리즘에 대해서 알아보려고 한다.
📌다익스트라
다익스트라(dijkstra) 알고리즘은 그래프에서 최단 경로를 구하는 알고리즘 중 하나이다. 다익스트라 알고리즘은 도착 정점 뿐만 아니라 하나의 정점에서 모든 다른 정점까지 최단 경로로 방문하며 각 정점까지의 최단 경로 를 모두 찾게 된다.
- 인접행렬을 사용: 정점의 수가 N일때, 인접행렬을 사용하면 각 단계마다 N개의 정점을 가중치가 짧은 간선을 찾기위해 순사탐색을 진행한다.
- 시간 복잡도: O(N²)
- 우선순위 큐 사용: 노드의 개수 V, 간선의 개수 E라고 했을때, 모든 간선을 확인하는 O(E)와 우선순위 큐에 최대로 들어갈 수 있는 수 O(E)와 추가 또는 삭제할때 드는 비용 O(LogE) -> 우선순위 큐에서의 시간 복잡도는 O(ELogE).
따라서 전체 시간 복잡도는 O(E + ELogE) = O(ELogE)이고, 보통 E <= V² 이므로, O(ELogV) 라고 볼 수 있다.
동작 과정(우선순위 큐)
- 모든 노드에 대한 거리를 무한대 값(충분히 큰 값)으로 초기화
- 시작 노드에 대한 거리를 0으로 초기화 후 우선순위 큐에 담음
- 우선순위 큐에서 가져온 노드에 대한 인접한(갈수 있는 노드)노드를 확인
- 더 빠르면 거리를 갱신하고 우선순위 큐에 넣어줌
- 방문 or 더 느리면 continue
- 우선순위 큐가 empty될때까지 3~5 반복
간단하게 1,2,3번 노드를 가지는 그래프에서 거리가 갱신되는 과정을 보면 아래와 같다.
코드(우선순위 큐)
인접한 정점들에 대한 list는 graph로 생성하였다고 가정
data class Node(
val end: Int,
val weight: Int
)
// 최단 경로 리스트
val distance = IntArray(6) { Int.MAX_VALUE }
fun dijkstra(start: Int) {
// 시작 노드의 거리는 0
distance[start] = 0
// 기준은 가중치로 오름차순
val pq = PriorityQueue<Node>(compareBy { it.weight })
// 시작 노드
pq.add(Node(start, 0))
while (pq.isNotEmpty()) {
val curr = pq.poll()
// 새로 선택된 node보다 더 가중치가 적음 -> 방문한 느드
if (distance[curr.end] < curr.weight) {
continue
}
// 현재 시작 노드의 인접 노드들 확인
for (node in graph[curr.end]) {
val nextNode = node.end
// 다음 노드까지의 거리 + 현재 노드까지의 거리 -> 방문후 가는 거리
val nextDistance = node.weight + curr.weight
// 지금까지의 거리 보다 방문후 가는 거리가 짧으면 업데이트 후 우선순위 큐에 넣어줌
if (nextDistance < distance[nextNode]) {
distance[nextNode] = nextDistance
pq.add(Node(nextNode, nextDistance))
}
}
}
}
📌벨만 포드
다익스트라 알고리즘과 마친가지로 _하나의 정점에서 모든 다른 정점까지 최단 경로로 방문하며 각 정점까지의 최단 거리 를 찾을 수 있지만 차이점은 음의 가중치가 있는 그래프 에서 사용할 수 있다.
벨만 포드 알고리즘에서는 음수 사이클 로 인하여 최단 거리를 정의할 수 없는 경우도 알아낼 수 있다.
1->2->3으로 가는경우 무한히 작아질 수 있으므로 최단 거리를 구할 수 없음 -> 계속 노드를 거쳐가면 1 + 2 -4 = -1 이므로 음의방향으로 발산한다.
벨만 포드 알고리즘과 다익스트라 알고리즘의 차이점은 음수 가중치 가 존재하는 경우에도 적용할 수 있다는 것이며, 시간 복잡도는 O(VE) (V: 정점 개수, E: 간선 대수)로 다익스트라에서 우선순위 큐를 이용한 방식인 O(ELogV) 보다 느리다.
동작 과정
- 모든 노드에 대한 거리를 무한대 값(충분히 큰 값)으로 초기화
- 다음의 과정을 V-1번 반복한다.
- 모든 간선 E개를 순서대로 확인한다.
- 각 간선을 가쳐가 다른 노드로 가는 비용을 비교하여 최단 거리로 갱신한다.
- 음수 간선 순환이 발생하는지 확인하기 위해서는 2번의 과정을 한 번 더 수행한다. -> 최단 거리가 갱신이 된다면 음수 간선 순환이 존재한다.
코드
fun bellmanFord(start: Int): Boolean {
// 최단 경로 리스트
val distance = IntArray(graph.size) { INF }
distance[start] = 0
for (i in graph.indices) {
for (j in graph.indices) {
if (graph[i][j] != INF) {
if (distance[j] > distance[i] + graph[i][j]) {
distance[j] = distance[i] + graph[i][j]
// N번째에서 갱신이 되었다면 음수 사이클 존재
if (i == graph.size - 1) {
return true
}
}
}
}
}
return false
}
📌플로이드 와샬
플로이드 와샬 알고리즘은 벨만포드와 같이 음의 가중치가 있는 그래프 에서 사용할 수 있고, 차이점은 하나의 정점이 아닌 모든 정점에서 모든 다른 정점으로 가는 최단 거리 를 구할 수 있다. 다시 말하여 다른 정점을 경유 하여 도착하는 최단 거리를 구할 수 있다.
플로이드-워셜 알고리즘은 DP(Dynamic Programming) 기법을 사용한 알고리즘이다. 각 정점으로 부터 다른 정점을 순차적으로 거쳐가면서 더 짧은 길이를 선택 하여 갱신 한다.
다익스트라와 다른점은 정점까지의 거리 dist 배열을 모든 정점 에서 모든 다른 정점으로 가는 최단 거리를 구해야 하므로 정점의 개수 V일때 V * V 크기의 dist 배열을 생성해주어야한다.
위의 거리 배열에서 각 dist[i][i]는 시작 정점으로 자기자신에 대한 거리는 0이므로 초기화 해준다.
동작 과정
- 인접행렬을 저장할 2차원 배열을 만들고 무한대(충분히 큰 값)로 초기화한다.
- 자기자신([i][i])는 0으로 초기화
- 그래프에 따라 간선 정보를 저장한다.
- 경유지를 거쳐가는 것이 빠르다면 거리 배열을 갱신 해준다.
- (시작: i, 도착: j일때 k가 경유 노드라면, dist[i][j] = min(dist[i][k] + dist[k][j])로 갱신)
- 모든 정점에 대해 순차적으로 경유지로 선택하여 2번 과정을 반복한다.
코드
fun floyd(dist: Array<IntArray>, n: Int) {
// k -> 거쳐가는 노드
for (k in 0 until n) {
// i -> 출발 노드
for (i in 0 until n) {
// j -> 도착 노드
for (j in 0 until n) {
dist[i][j] = dist[i][j].coerceAtMost(dist[i][k] + dist[k][j])
}
}
}
}