Skip to content

Commit abc274a

Browse files
committed
stack and depth first search
1 parent 4aeff37 commit abc274a

File tree

4 files changed

+195
-1
lines changed

4 files changed

+195
-1
lines changed

README.md

+24-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,27 @@
11
# js-graph-algorithms
22
Package provides javascript implementation of algorithms for graph processing
33

4-
[![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)
4+
[![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)
5+
6+
# Usage
7+
8+
### Create an undirected unweighted graph
9+
10+
The sample code below shows how to create a undirected and unweighted graph:
11+
12+
```javascript
13+
jsgraphs = require('js-graph-algorithms');
14+
15+
var g = new jsgraphs.Graph(6);
16+
g.addEdge(0, 5); // add undirected edge connecting vertex 0 to vertex 5
17+
g.addEdge(2, 4);
18+
g.addEdge(2, 3);
19+
g.addEdge(1, 2);
20+
g.addEdge(0, 1);
21+
g.addEdge(3, 4);
22+
g.addEdge(3, 5);
23+
g.addEdge(0, 2);
24+
25+
console.log(g.V); // display 6, which is the number of vertices in g
26+
console.log(g.adj(0)); // display [5, 1, 2], which is the adjacent list to vertex 0
27+
```

src/jsgraphs.js

+111
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,75 @@
11
var jsgraphs = jsgraphs || {};
22

33
(function(jss){
4+
var StackNode = function (value) {
5+
this.value = value;
6+
this.next = null;
7+
};
8+
9+
jss.StackNode = StackNode;
10+
11+
var Stack = function() {
12+
this.N = 0;
13+
this.first = null;
14+
};
15+
16+
Stack.prototype.push = function (a) {
17+
this.first = this._push(this.first, a);
18+
};
19+
20+
Stack.prototype._push = function(x, a) {
21+
if(x == null) {
22+
this.N++;
23+
return new jss.StackNode(a);
24+
}
25+
var oldX = x;
26+
this.N++;
27+
x = new jss.StackNode(a);
28+
x.next = oldX;
29+
return x;
30+
};
31+
32+
Stack.prototype.pop = function () {
33+
if (this.first == null) {
34+
return undefined;
35+
}
36+
37+
var oldFirst = this.first;
38+
var item = oldFirst.value;
39+
this.first = oldFirst.next;
40+
this.N--;
41+
42+
return item;
43+
};
44+
45+
Stack.prototype.size = function() {
46+
return this.N;
47+
};
48+
49+
Stack.prototype.isEmpty = function() {
50+
return this.N == 0;
51+
};
52+
53+
Stack.prototype.peep = function() {
54+
if (this.first == null) {
55+
return undefined;
56+
}
57+
58+
return this.first.value;
59+
};
60+
61+
Stack.prototype.toArray = function() {
62+
var result = [];
63+
x = this.first;
64+
while (x != null) {
65+
result.push(x.value);
66+
x = x.next;
67+
}
68+
return result;
69+
};
70+
71+
jss.Stack = Stack;
72+
473
var Graph = function (V) {
574
this.V = V;
675
this.adjList = [];
@@ -19,6 +88,48 @@ var jsgraphs = jsgraphs || {};
1988
};
2089

2190
jss.Graph = Graph;
91+
92+
var DepthFirstSearch = function(G, s) {
93+
this.s = s;
94+
var V = G.V;
95+
this.marked = [];
96+
this.edgeTo = [];
97+
for (var v = 0; v < V; ++v) {
98+
this.marked.push(false);
99+
this.edgeTo.push(-1);
100+
}
101+
102+
this.dfs(G, s);
103+
};
104+
105+
DepthFirstSearch.prototype.dfs = function (G, v) {
106+
this.marked[v] = true;
107+
var adj_v = G.adj(v);
108+
for (var i = 0; i < adj_v.length; ++i){
109+
var w = adj_v[i];
110+
if (!this.marked[w]){
111+
this.edgeTo[w] = v;
112+
this.dfs(G, w);
113+
}
114+
}
115+
};
116+
117+
DepthFirstSearch.prototype.hasPathTo = function(v) {
118+
return this.marked[v];
119+
};
120+
121+
DepthFirstSearch.prototype.pathTo = function(v) {
122+
var path = new jss.Stack();
123+
if(v == this.s) return [v];
124+
125+
for(var x = v; x != this.s; x = this.edgeTo[x]) {
126+
path.push(x);
127+
}
128+
path.push(this.s);
129+
return path.toArray();
130+
};
131+
132+
jss.DepthFirstSearch = DepthFirstSearch;
22133

23134
})(jsgraphs);
24135

test/depth-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 dfs = new jsgraphs.DepthFirstSearch(g, s);
17+
18+
19+
for(var v=0; v < g.V; ++v) {
20+
if(dfs.hasPathTo(v)) {
21+
console.log(s + " is connected to " + v);
22+
console.log("path: " + dfs.pathTo(v));
23+
} else {
24+
console.log('No path from ' + s + ' to ' + v);
25+
}
26+
}
27+
});
28+
});

test/stack-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('Stack operations', function(){
5+
describe('Stack push and pop', function(){
6+
7+
8+
it('should be able to push, peep and pop', function(){
9+
var stack = new jsgraphs.Stack();
10+
stack.push(10);
11+
expect(stack.size()).to.equal(1);
12+
expect(stack.peep()).to.equal(10);
13+
expect(stack.pop()).to.equal(10);
14+
expect(stack.size()).to.equal(0);
15+
expect(stack.isEmpty()).to.equal(true);
16+
for(var i = 0; i < 100; ++i) {
17+
stack.push(i);
18+
expect(stack.size()).to.equal(i+1);
19+
}
20+
var a = stack.toArray();
21+
for(var i = 0; i < 100; ++i) {
22+
expect(a[i]).to.equal(100 - i - 1);
23+
}
24+
25+
for(var i = 0; i < 100; ++i) {
26+
expect(stack.pop()).to.equal(100 - i - 1);
27+
expect(stack.size()).to.equal(100 -i - 1);
28+
}
29+
30+
});
31+
});
32+
});

0 commit comments

Comments
 (0)