Skip to content

Commit 79c75c4

Browse files
committed
flow network
1 parent 04cb5d0 commit 79c75c4

File tree

4 files changed

+151
-0
lines changed

4 files changed

+151
-0
lines changed

README.md

+6
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,12 @@ Package provides javascript implementation of algorithms for graph processing
1414
* Shortest Paths (Dijkstra, Bellman-Ford, Topological Sort on DAG)
1515
* MaxFlow-MinCut (Ford-Fulkerson)
1616

17+
# Install
18+
19+
```bash
20+
npm install js-graph-algorithms
21+
```
22+
1723
# Usage
1824

1925
### Create an undirected unweighted graph

src/jsgraphs.js

+113
Original file line numberDiff line numberDiff line change
@@ -463,6 +463,65 @@ var jsgraphs = jsgraphs || {};
463463

464464
jss.WeightedDiGraph = WeightedDiGraph;
465465

466+
var FlowEdge = function(v, w, capacity) {
467+
this.v = v;
468+
this.w = w;
469+
this.capacity = capacity;
470+
this.flow = 0;
471+
};
472+
473+
FlowEdge.prototype.residualCapacityTo = function (x) {
474+
if(x == this.v) {
475+
return this.flow;
476+
} else {
477+
return this.capacity - this.flow;
478+
}
479+
};
480+
481+
FlowEdge.prototype.addResidualFlowTo = function (x, deltaFlow) {
482+
if(x == this.v) {
483+
this.flow -= deltaFlow;
484+
} else if(x == this.w) {
485+
this.flow += deltaFlow;
486+
}
487+
};
488+
489+
FlowEdge.prototype.from = function() {
490+
return this.v;
491+
};
492+
493+
FlowEdge.prototype.to = function() {
494+
return this.w;
495+
};
496+
497+
FlowEdge.prototype.other = function(x) {
498+
return x == this.v ? this.w : this.v;
499+
}
500+
501+
502+
jss.FlowEdge = FlowEdge;
503+
504+
var FlowNetwork = function(V) {
505+
this.V = V;
506+
this.adjList = [];
507+
for(var v = 0; v < V; ++v) {
508+
this.adjList.push([]);
509+
}
510+
};
511+
512+
FlowNetwork.prototype.addEdge = function(e) {
513+
var v = e.from();
514+
this.adjList[v].push(e);
515+
var w = e.other(v);
516+
this.adjList[w].push(e);
517+
};
518+
519+
FlowNetwork.prototype.adj = function(v) {
520+
return this.adjList[v];
521+
};
522+
523+
jss.FlowNetwork = FlowNetwork;
524+
466525
var DepthFirstSearch = function(G, s) {
467526
this.s = s;
468527
var V = G.V;
@@ -980,6 +1039,60 @@ var jsgraphs = jsgraphs || {};
9801039
};
9811040

9821041
jss.TopologicalSortShortestPaths = TopologicalSortShortestPaths;
1042+
1043+
var FordFulkerson = function(G, s, t) {
1044+
this.value = 0;
1045+
var V = G.V;
1046+
var bottle = Numbers.MAX_VALUE;
1047+
this.marked = null;
1048+
this.edgeTo = null;
1049+
this.s = s;
1050+
this.t = t;
1051+
while(this.hasAugmentedPath(G)){
1052+
1053+
for(var x = this.t; x != this.s; x = this.edgeTo[x].from()) {
1054+
bottle = Math.min(bottle, this.edgeTo[x].residualCapacityTo(x));
1055+
}
1056+
1057+
for(var x = this.t; x != this.s; x = this.edgeTo[x].from()) {
1058+
this.edgeTo[x].addResidualFlowTo(x, bottle);
1059+
}
1060+
1061+
this.value += bottle;
1062+
}
1063+
};
1064+
1065+
FordFulkerson.prototype.hasAugmentedPath = function(G) {
1066+
var V = G.V;
1067+
this.marked = [];
1068+
this.edgeTo = [];
1069+
for(var v = 0; v < V; ++v) {
1070+
this.marked.push(false);
1071+
this.edgeTo.push(null);
1072+
}
1073+
1074+
var queue = new jss.Queue();
1075+
queue.push(this.s);
1076+
1077+
while(!queue.isEmpty()){
1078+
var v = queue.dequeue();
1079+
var adj_v = G.adj(v);
1080+
this.marked[v] = true;
1081+
for (var i = 0; i < adj_v.length; ++i) {
1082+
var e = adj_v[i];
1083+
var w = e.other(v);
1084+
if(e.residualCapacityTo(v) > 0){
1085+
if(w == this.t){
1086+
return true;
1087+
}
1088+
this.edgeTo[w] = e;
1089+
queue.push(w);
1090+
}
1091+
}
1092+
}
1093+
1094+
return false;
1095+
};
9831096

9841097
})(jsgraphs);
9851098

test/flow-network-spec.js

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
var expect = require('chai').expect;
2+
var jsgraphs = require('../src/jsgraphs');
3+
4+
describe('Flow Network', function(){
5+
it('should confirms to flow edge spec', function(){
6+
var g = new jsgraphs.FlowNetwork(8);
7+
g.addEdge(new jsgraphs.FlowEdge(0, 1, 10));
8+
g.addEdge(new jsgraphs.FlowEdge(0, 2, 5));
9+
g.addEdge(new jsgraphs.FlowEdge(0, 3, 15));
10+
g.addEdge(new jsgraphs.FlowEdge(1, 4, 9));
11+
g.addEdge(new jsgraphs.FlowEdge(1, 5, 15));
12+
g.addEdge(new jsgraphs.FlowEdge(1, 2, 4));
13+
g.addEdge(new jsgraphs.FlowEdge(2, 5, 8));
14+
g.addEdge(new jsgraphs.FlowEdge(2, 3, 4));
15+
g.addEdge(new jsgraphs.FlowEdge(3, 6, 16));
16+
g.addEdge(new jsgraphs.FlowEdge(4, 5, 15));
17+
g.addEdge(new jsgraphs.FlowEdge(4, 7, 10));
18+
g.addEdge(new jsgraphs.FlowEdge(5, 7, 10));
19+
g.addEdge(new jsgraphs.FlowEdge(5, 6, 15));
20+
g.addEdge(new jsgraphs.FlowEdge(6, 2, 6));
21+
g.addEdge(new jsgraphs.FlowEdge(6, 7, 10));
22+
23+
expect(g.V).to.equal(8);
24+
25+
var edgeCount = 0;
26+
for(var v = 0; v < g.V; ++v) {
27+
var adj_v = g.adj(v);
28+
edgeCount += adj_v.length;
29+
}
30+
expect(edgeCount).to.equal(30);
31+
});
32+
});

test/ford-fulkerson-spec.js

Whitespace-only changes.

0 commit comments

Comments
 (0)