mirror of
https://hub.njuu.cf/TheAlgorithms/Python.git
synced 2023-10-11 13:06:12 +08:00
I've added comments to make understanding this method a little easier for those that are not familiar. This should close out #727 . Other changes: 1. added if __name__ == '__main__' rather than the "# MAIN" comment 2. put in return for distances and vertices. Previously everything was just printed out, but someone may find it useful to have the algorithm return something. 3. Other PEP8 changes 4. Added example input and expected output as a check to make sure any future changes will give the same output.
This commit is contained in:
parent
b8937364dc
commit
a99acae32d
@ -1,9 +1,16 @@
|
|||||||
|
# floyd_warshall.py
|
||||||
|
"""
|
||||||
|
The problem is to find the shortest distance between all pairs of vertices in a weighted directed graph that can
|
||||||
|
have negative edge weights.
|
||||||
|
"""
|
||||||
|
|
||||||
from __future__ import print_function
|
from __future__ import print_function
|
||||||
|
|
||||||
def printDist(dist, V):
|
|
||||||
|
def _print_dist(dist, v):
|
||||||
print("\nThe shortest path matrix using Floyd Warshall algorithm\n")
|
print("\nThe shortest path matrix using Floyd Warshall algorithm\n")
|
||||||
for i in range(V):
|
for i in range(v):
|
||||||
for j in range(V):
|
for j in range(v):
|
||||||
if dist[i][j] != float('inf') :
|
if dist[i][j] != float('inf') :
|
||||||
print(int(dist[i][j]),end = "\t")
|
print(int(dist[i][j]),end = "\t")
|
||||||
else:
|
else:
|
||||||
@ -12,37 +19,84 @@ def printDist(dist, V):
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
def FloydWarshall(graph, V):
|
def floyd_warshall(graph, v):
|
||||||
dist=[[float('inf') for i in range(V)] for j in range(V)]
|
"""
|
||||||
|
:param graph: 2D array calculated from weight[edge[i, j]]
|
||||||
|
:type graph: List[List[float]]
|
||||||
|
:param v: number of vertices
|
||||||
|
:type v: int
|
||||||
|
:return: shortest distance between all vertex pairs
|
||||||
|
distance[u][v] will contain the shortest distance from vertex u to v.
|
||||||
|
|
||||||
for i in range(V):
|
1. For all edges from v to n, distance[i][j] = weight(edge(i, j)).
|
||||||
for j in range(V):
|
3. The algorithm then performs distance[i][j] = min(distance[i][j], distance[i][k] + distance[k][j]) for each
|
||||||
|
possible pair i, j of vertices.
|
||||||
|
4. The above is repeated for each vertex k in the graph.
|
||||||
|
5. Whenever distance[i][j] is given a new minimum value, next vertex[i][j] is updated to the next vertex[i][k].
|
||||||
|
"""
|
||||||
|
|
||||||
|
dist=[[float('inf') for _ in range(v)] for _ in range(v)]
|
||||||
|
|
||||||
|
for i in range(v):
|
||||||
|
for j in range(v):
|
||||||
dist[i][j] = graph[i][j]
|
dist[i][j] = graph[i][j]
|
||||||
|
|
||||||
for k in range(V):
|
# check vertex k against all other vertices (i, j)
|
||||||
for i in range(V):
|
for k in range(v):
|
||||||
for j in range(V):
|
# looping through rows of graph array
|
||||||
|
for i in range(v):
|
||||||
|
# looping through columns of graph array
|
||||||
|
for j in range(v):
|
||||||
if dist[i][k]!=float('inf') and dist[k][j]!=float('inf') and dist[i][k]+dist[k][j] < dist[i][j]:
|
if dist[i][k]!=float('inf') and dist[k][j]!=float('inf') and dist[i][k]+dist[k][j] < dist[i][j]:
|
||||||
dist[i][j] = dist[i][k] + dist[k][j]
|
dist[i][j] = dist[i][k] + dist[k][j]
|
||||||
|
|
||||||
printDist(dist, V)
|
_print_dist(dist, v)
|
||||||
|
return dist, v
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#MAIN
|
if __name__== '__main__':
|
||||||
V = int(input("Enter number of vertices: "))
|
v = int(input("Enter number of vertices: "))
|
||||||
E = int(input("Enter number of edges: "))
|
e = int(input("Enter number of edges: "))
|
||||||
|
|
||||||
graph = [[float('inf') for i in range(V)] for j in range(V)]
|
graph = [[float('inf') for i in range(v)] for j in range(v)]
|
||||||
|
|
||||||
for i in range(V):
|
for i in range(v):
|
||||||
graph[i][i] = 0.0
|
graph[i][i] = 0.0
|
||||||
|
|
||||||
for i in range(E):
|
# src and dst are indices that must be within the array size graph[e][v]
|
||||||
print("\nEdge ",i+1)
|
# failure to follow this will result in an error
|
||||||
src = int(input("Enter source:"))
|
for i in range(e):
|
||||||
dst = int(input("Enter destination:"))
|
print("\nEdge ",i+1)
|
||||||
weight = float(input("Enter weight:"))
|
src = int(input("Enter source:"))
|
||||||
graph[src][dst] = weight
|
dst = int(input("Enter destination:"))
|
||||||
|
weight = float(input("Enter weight:"))
|
||||||
|
graph[src][dst] = weight
|
||||||
|
|
||||||
FloydWarshall(graph, V)
|
floyd_warshall(graph, v)
|
||||||
|
|
||||||
|
|
||||||
|
# Example Input
|
||||||
|
# Enter number of vertices: 3
|
||||||
|
# Enter number of edges: 2
|
||||||
|
|
||||||
|
# # generated graph from vertex and edge inputs
|
||||||
|
# [[inf, inf, inf], [inf, inf, inf], [inf, inf, inf]]
|
||||||
|
# [[0.0, inf, inf], [inf, 0.0, inf], [inf, inf, 0.0]]
|
||||||
|
|
||||||
|
# specify source, destination and weight for edge #1
|
||||||
|
# Edge 1
|
||||||
|
# Enter source:1
|
||||||
|
# Enter destination:2
|
||||||
|
# Enter weight:2
|
||||||
|
|
||||||
|
# specify source, destination and weight for edge #2
|
||||||
|
# Edge 2
|
||||||
|
# Enter source:2
|
||||||
|
# Enter destination:1
|
||||||
|
# Enter weight:1
|
||||||
|
|
||||||
|
# # Expected Output from the vertice, edge and src, dst, weight inputs!!
|
||||||
|
# 0 INF INF
|
||||||
|
# INF 0 2
|
||||||
|
# INF 1 0
|
||||||
|
Loading…
Reference in New Issue
Block a user