Skip to content

Commit 1668afe

Browse files
committed
breadth first search
1 parent abc274a commit 1668afe

File tree

4 files changed

+196
-0
lines changed

4 files changed

+196
-0
lines changed

README.md

+38
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,14 @@ Package provides javascript implementation of algorithms for graph processing
33

44
[![Build Status](https://travis-ci.org/chen0040/js-graph-algorithms.svg?branch=master)](https://travis-ci.org/chen0040/js-graph-algorithms) [![Coverage Status](https://coveralls.io/repos/github/chen0040/js-graph-algorithms/badge.svg?branch=master)](https://coveralls.io/github/chen0040/js-graph-algorithms?branch=master)
55

6+
# Features
7+
8+
* Depth First Search
9+
* Breadth First Search
10+
* Minimum Spanning Tree (Kruskal, Prim Lazy, Prim Eager)
11+
* Shortest Paths (Dijkstra, Bellman-Ford, Topological Sort on DAG)
12+
* MaxFlow-MinCut (Ford-Fulkerson)
13+
614
# Usage
715

816
### Create an undirected unweighted graph
@@ -24,4 +32,34 @@ g.addEdge(0, 2);
2432

2533
console.log(g.V); // display 6, which is the number of vertices in g
2634
console.log(g.adj(0)); // display [5, 1, 2], which is the adjacent list to vertex 0
35+
```
36+
37+
### Depth First Search
38+
39+
The sample code below show how to perform depth first search of an undirected graph
40+
41+
```javascript
42+
jsgraphs = require('js-graph-algorithms');
43+
44+
var g = new jsgraphs.Graph(6);
45+
g.addEdge(0, 5);
46+
g.addEdge(2, 4);
47+
g.addEdge(2, 3);
48+
g.addEdge(1, 2);
49+
g.addEdge(0, 1);
50+
g.addEdge(3, 4);
51+
g.addEdge(3, 5);
52+
g.addEdge(0, 2);
53+
var s = 0;
54+
var dfs = new jsgraphs.DepthFirstSearch(g, s);
55+
56+
57+
for(var v=0; v < g.V; ++v) {
58+
if(dfs.hasPathTo(v)) {
59+
console.log(s + " is connected to " + v);
60+
console.log("path: " + dfs.pathTo(v));
61+
} else {
62+
console.log('No path from ' + s + ' to ' + v);
63+
}
64+
}
2765
```

src/jsgraphs.js

+108
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,69 @@ var jsgraphs = jsgraphs || {};
7070

7171
jss.Stack = Stack;
7272

73+
var QueueNode = function(a) {
74+
this.value = a;
75+
this.next = null;
76+
};
77+
78+
jss.QueueNode = QueueNode;
79+
80+
var Queue = function() {
81+
this.first = null;
82+
this.last = null;
83+
this.N = 0;
84+
};
85+
86+
Queue.prototype.enqueue = function(item) {
87+
var oldLast = this.last;
88+
this.last = new jss.QueueNode(item);
89+
if(oldLast != null){
90+
oldLast.next = this.last;
91+
}
92+
if (this.first == null) {
93+
this.first = this.last;
94+
}
95+
this.N++;
96+
};
97+
98+
Queue.prototype.dequeue = function() {
99+
if(this.first == null) {
100+
return undefined;
101+
}
102+
103+
var oldFirst = this.first;
104+
var item = oldFirst.value;
105+
this.first = oldFirst.next;
106+
107+
if(this.first == null) {
108+
this.last = null;
109+
}
110+
111+
this.N--;
112+
113+
return item;
114+
};
115+
116+
Queue.prototype.size = function() {
117+
return this.N;
118+
};
119+
120+
Queue.prototype.isEmpty = function() {
121+
return this.N == 0;
122+
};
123+
124+
Queue.prototype.toArray = function() {
125+
var result = [];
126+
var x = this.first;
127+
while (x != null) {
128+
result.push(x.value);
129+
x = x.next;
130+
}
131+
return result;
132+
};
133+
134+
jss.Queue = Queue;
135+
73136
var Graph = function (V) {
74137
this.V = V;
75138
this.adjList = [];
@@ -130,6 +193,51 @@ var jsgraphs = jsgraphs || {};
130193
};
131194

132195
jss.DepthFirstSearch = DepthFirstSearch;
196+
197+
var BreadthFirstSearch = function(G, s) {
198+
var V = G.V;
199+
this.s = s;
200+
201+
var queue = new jss.Queue();
202+
queue.enqueue(s);
203+
this.marked = [];
204+
this.edgeTo = [];
205+
206+
for(var v = 0; v < V; ++v) {
207+
this.marked.push(false);
208+
this.edgeTo.push(-1);
209+
}
210+
211+
while (!queue.isEmpty()) {
212+
var v = queue.dequeue();
213+
this.marked[v] = true;
214+
var adj_v = G.adj(v);
215+
for (var i = 0; i < adj_v.length; ++i) {
216+
var w = adj_v[i];
217+
if(!this.marked[w]){
218+
this.edgeTo[w] = v;
219+
queue.enqueue(w);
220+
}
221+
}
222+
}
223+
};
224+
225+
BreadthFirstSearch.prototype.hasPathTo = function(v) {
226+
return this.marked[v];
227+
};
228+
229+
BreadthFirstSearch.prototype.pathTo = function(v) {
230+
var path = new jss.Stack();
231+
if(v == this.s) return [v];
232+
233+
for(var x = v; x != this.s; x = this.edgeTo[x]) {
234+
path.push(x);
235+
}
236+
path.push(this.s);
237+
return path.toArray();
238+
};
239+
240+
jss.BreadthFirstSearch = BreadthFirstSearch;
133241

134242
})(jsgraphs);
135243

test/breadth-first-search-spec.js

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
var expect = require('chai').expect;
2+
var jsgraphs = require('../src/jsgraphs');
3+
4+
describe('Depth first search', function(){
5+
it('should traverse correctly', function(){
6+
var g = new jsgraphs.Graph(6);
7+
g.addEdge(0, 5);
8+
g.addEdge(2, 4);
9+
g.addEdge(2, 3);
10+
g.addEdge(1, 2);
11+
g.addEdge(0, 1);
12+
g.addEdge(3, 4);
13+
g.addEdge(3, 5);
14+
g.addEdge(0, 2);
15+
var s = 0;
16+
var bfs = new jsgraphs.BreadthFirstSearch(g, s);
17+
18+
19+
for(var v=0; v < g.V; ++v) {
20+
if(bfs.hasPathTo(v)) {
21+
console.log(s + " is connected to " + v);
22+
console.log("path: " + bfs.pathTo(v));
23+
} else {
24+
console.log('No path from ' + s + ' to ' + v);
25+
}
26+
}
27+
});
28+
});

test/queue-spec.js

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
var expect = require('chai').expect;
2+
var jsgraphs = require('../src/jsgraphs');
3+
4+
describe('Queue operations', function() {
5+
it('should enqueue and dequeue correctly', function() {
6+
var queue = new jsgraphs.Queue();
7+
queue.enqueue(10);
8+
queue.enqueue(20);
9+
queue.enqueue(30);
10+
var a = queue.toArray();
11+
expect(a.length).to.equal(3);
12+
for(var i=0; i < a.length; ++i) {
13+
expect(a[i]).to.equal((i+1) * 10);
14+
}
15+
expect(queue.size()).to.equal(3);
16+
expect(queue.isEmpty()).to.equal(false);
17+
expect(queue.dequeue()).to.equal(10);
18+
expect(queue.dequeue()).to.equal(20);
19+
expect(queue.dequeue()).to.equal(30);
20+
expect(queue.isEmpty()).to.equal(true);
21+
});
22+
});

0 commit comments

Comments
 (0)