Skip to content

Commit 032a375

Browse files
committed
feat: add edmond-karp algorithm for max-flow
1 parent a938a23 commit 032a375

File tree

3 files changed

+173
-0
lines changed

3 files changed

+173
-0
lines changed

README.md

+1
Original file line numberDiff line numberDiff line change
@@ -462,6 +462,7 @@ Read our [Contribution Guidelines](CONTRIBUTING.md) before you contribute.
462462
12. [`NewUnionFind`](./graph/unionfind.go#L24): Initialise a new union find data structure with s nodes
463463
13. [`NotExist`](./graph/depthfirstsearch.go#L12): No description provided.
464464
14. [`Topological`](./graph/topological.go#L7): Topological assumes that graph given is valid and that its possible to get a topological ordering. constraints are array of []int{a, b}, representing an edge going from a to b
465+
15. [`Edmond-Karp`](./graph/edmondkarp.go#L43): Computes the maximum possible flow between a pair of s-t vertices in a weighted graph
465466

466467
---
467468
##### Types

graph/edmondkarp.go

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
// Edmond-Karp algorithm is an implementation of the Ford-Fulkerson method
2+
// to compute max-flow between a pair of source-sink vertices in a weighted graph
3+
// It uses BFS (Breadth First Search) to find the residual paths
4+
// Time Complexity: O(V * E^2) where V is the number of vertices and E is the number of edges
5+
// Space Complexity: O(V + E) Because we keep residual graph in size of the original graph
6+
// Thomas H. Cormen, Charles E. Leiserson, Ronald L. Rivest, and Clifford Stein. 2009. Introduction to Algorithms, Third Edition (3rd. ed.). The MIT Press.
7+
8+
package graph
9+
10+
import (
11+
"math"
12+
)
13+
14+
// Returns a mapping of vertices as path, if there is any from source to sink
15+
// Otherwise, returns nil
16+
func FindPath(rGraph WeightedGraph, source int, sink int) map[int]int {
17+
queue := make([]int, 0)
18+
marked := make([]bool, len(rGraph))
19+
marked[source] = true
20+
queue = append(queue, source)
21+
parent := make(map[int]int)
22+
23+
// BFS loop with saving the path found
24+
for len(queue) > 0 {
25+
v := queue[0]
26+
queue = queue[1:]
27+
for i := 0; i < len(rGraph[v]); i++ {
28+
if !marked[i] && rGraph[v][i] > 0 {
29+
parent[i] = v
30+
// Terminate the BFS, if we reach to sink
31+
if i == sink {
32+
return parent
33+
}
34+
marked[i] = true
35+
queue = append(queue, i)
36+
}
37+
}
38+
}
39+
// source and sink are not in the same connected component
40+
return nil
41+
}
42+
43+
func EdmondKarp(graph WeightedGraph, source int, sink int) float64 {
44+
// Check graph emptiness
45+
if len(graph) == 0 {
46+
return 0.0
47+
}
48+
49+
// Check correct dimensions of the graph slice
50+
for i := 0; i < len(graph); i++ {
51+
if len(graph[i]) != len(graph) {
52+
return 0.0
53+
}
54+
}
55+
56+
rGraph := make(WeightedGraph, len(graph))
57+
for i := 0; i < len(graph); i++ {
58+
rGraph[i] = make([]float64, len(graph))
59+
}
60+
// Init the residual graph with the same capacities as the original graph
61+
copy(rGraph, graph)
62+
63+
maxFlow := 0.0
64+
65+
for {
66+
parent := FindPath(rGraph, source, sink)
67+
if parent == nil {
68+
break
69+
}
70+
// Finding the max flow over the path returned by BFS
71+
// i.e. finding minimum residual capacity amonth the path edges
72+
pathFlow := math.MaxFloat64
73+
for v := sink; v != source; v = parent[v] {
74+
u := parent[v]
75+
if rGraph[u][v] < pathFlow {
76+
pathFlow = rGraph[u][v]
77+
}
78+
}
79+
80+
// update residual capacities of the edges and
81+
// reverse edges along the path
82+
for v := sink; v != source; v = parent[v] {
83+
u := parent[v]
84+
rGraph[u][v] -= pathFlow
85+
rGraph[v][u] += pathFlow
86+
}
87+
88+
// Update the total flow found so far
89+
maxFlow += pathFlow
90+
}
91+
92+
return maxFlow
93+
}

graph/edmondkarp_test.go

+79
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
package graph
2+
3+
import (
4+
"testing"
5+
)
6+
7+
func TestEdmondKarp(t *testing.T) {
8+
var edmondKarpTestData = []struct {
9+
description string
10+
graph WeightedGraph
11+
source int
12+
sink int
13+
expected float64
14+
}{
15+
{
16+
description: "test empty graph",
17+
graph: nil,
18+
source: 0,
19+
sink: 0,
20+
expected: 0.0,
21+
},
22+
{
23+
description: "test graph with wrong dimensions",
24+
graph: WeightedGraph{
25+
{1, 2},
26+
{0},
27+
},
28+
source: 0,
29+
sink: 1,
30+
expected: 0.0,
31+
},
32+
{
33+
description: "test graph with no edges",
34+
graph: WeightedGraph{
35+
{0, 0},
36+
{0, 0},
37+
},
38+
source: 0,
39+
sink: 1,
40+
expected: 0.0,
41+
},
42+
{
43+
description: "test graph with 4 vertices",
44+
graph: WeightedGraph{
45+
{0, 1000000, 1000000, 0},
46+
{0, 0, 1, 1000000},
47+
{0, 0, 0, 1000000},
48+
{0, 0, 0, 0},
49+
},
50+
source: 0,
51+
sink: 3,
52+
expected: 2000000,
53+
},
54+
{
55+
description: "test graph with 6 vertices and some float64 weights",
56+
graph: WeightedGraph{
57+
{0, 16, 13.8, 0, 0, 0},
58+
{0, 0, 10, 12.7, 0, 0},
59+
{0, 4.2, 0, 0, 14, 0},
60+
{0, 0, 9.1, 0, 0, 21.3},
61+
{0, 0, 0, 7.5, 0, 4},
62+
{0, 0, 0, 0, 0, 0},
63+
},
64+
source: 0,
65+
sink: 5,
66+
expected: 24.2,
67+
},
68+
}
69+
for _, test := range edmondKarpTestData {
70+
t.Run(test.description, func(t *testing.T) {
71+
result := EdmondKarp(test.graph, test.source, test.sink)
72+
73+
if !almostEqual(test.expected, result) {
74+
t.Logf("FAIL: %s", test.description)
75+
t.Fatalf("Expected result:%f\nFound: %f", test.expected, result)
76+
}
77+
})
78+
}
79+
}

0 commit comments

Comments
 (0)