-
Notifications
You must be signed in to change notification settings - Fork 2.1k
/
Copy pathFord_Fulkerson_Method.js
142 lines (136 loc) · 3.86 KB
/
Ford_Fulkerson_Method.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
//+++++++++++++++Utility+++++++++++++++++
//Queue Data Structure implemented
function Queue() {
this.elements = [];
}
Queue.prototype.enqueue = function (e) {
this.elements.push(e);
};
// remove an element from the front of the queue
Queue.prototype.dequeue = function () {
return this.elements.shift();
};
// check if the queue is empty
Queue.prototype.isEmpty = function () {
return this.elements.length == 0;
};
// get the element at the front of the queue
Queue.prototype.peek = function () {
return !this.isEmpty() ? this.elements[0] : undefined;
};
Queue.prototype.length = function() {
return this.elements.length;
}
//++++++++++++++++++++++++++++++++++++++++++
//Function to do breadth first search in graph
function bfs(reducedGraph, source, sink, path, V)
{
//array to store if the node is visited or not
var visited = [V];
//intially no node is visited
for (var i = 0; i < V; i++)
visited[i] = false;
let q = new Queue();
//Starting from source
q.enqueue(source);
visited[source] = true;
//parent of starting node will be nothing hence -1 is stored in it
path[source] = -1;
while (!q.isEmpty())
{
var u = q.peek();
q.dequeue();
for(var v = 0; v < V; v++)
{
if(visited[v] === false && reducedGraph[u][v] > 0)
{
q.enqueue(v);
path[v] = u;
visited[v] = true;
}
}
}
//If there exist path from source to sink then it will return true else false
return (visited[sink] === true);
}
function fordFulkerson(graph,source,sink,V)
{
//reducedGraph is used to store the remaining capacity. reducedGraph[i][j]
//indicates remaining capacity of edge from i to jif edge exists.
var reducedGraph = [];
for( var i = 0; i < V; i++)
{
reducedGraph[i] = [];
for(var j = 0; j < V; j++)
{
reducedGraph[i][j] = graph[i][j];
}
}
//Path is used to store parent
var path = [V];
//variable to store maximum flow in the path chosen
var max_flow = 0;
while (bfs(reducedGraph, source, sink, path, V))
{
var path_flow = Number.MAX_VALUE;
//Finding maximum flow in the path selected
for(var v = sink; v != source; v = path[v])
{
var u = path[v];
path_flow = Math.min( path_flow, reducedGraph[u][v]);
}
//Updating remaining capacity odf the edges
for( var v = sink ; v != source; v = path[v])
{
var u = path[v];
reducedGraph[u][v] -= path_flow;
reducedGraph[v][u] += path_flow;
}
max_flow += path_flow;
}
return max_flow;
}
//Used synchronous readline package to take console input
var readlineSync = require('readline-sync');
//Input number of nodes in a graph
var V = readlineSync.question('Enter number of nodes in a graph: ');
//Input the weight matrix for the directed graph
var graph = [];
for(var i = 0; i < V; i++)
{
graph[i] = [];
for(var j = 0; j < V; j++)
{
console.log(`Enter the weight form ` , i , `to `,j, `: `);
var a = readlineSync.question();
graph[i][j] = a;
}
}
var source = readlineSync.question("Enter source: ");
var sink = readlineSync.question("Enter Sink: ");
var ans = fordFulkerson(graph,source,sink,V);
console.log('Maximum flow from source to sink is: ', ans);
/*Input1:
V = 6
graph = [ [0, 16, 13, 0, 0, 0],
[0, 0, 10, 12, 0, 0],
[0, 4, 0, 0, 14, 0],
[0, 0, 9, 0, 0, 20],
[0, 0, 0, 7, 0, 4],
[0, 0, 0, 0, 0, 0]
];
source = 0 sink = 5
Output1: 23
Input2:
V = 6
graph = [ [0, 10, 0, 3, 0, 0],
[0, 0, 8, 2, 0, 0],
[0, 0, 0, 0, , 10],
[0, 0, 6, 0, 7, 0],
[0, 0, 0, 0, 10, 0],
[0, 0, 0, 0, 0, 0]
];
source = 0
sink = 5
Output2: 10
*/