Skip to content

Commit 47ac5fc

Browse files
committed
Add detect cycle.
1 parent 843893e commit 47ac5fc

File tree

8 files changed

+192
-2
lines changed

8 files changed

+192
-2
lines changed

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@
6868
* [Breadth-First Search](https://github.com/trekhleb/javascript-algorithms/tree/master/src/algorithms/graph/breadth-first-search) (BFS)
6969
* [Dijkstra Algorithm](https://github.com/trekhleb/javascript-algorithms/tree/master/src/algorithms/graph/dijkstra) - finding shortest path to all graph vertices
7070
* [Bellman-Ford Algorithm](https://github.com/trekhleb/javascript-algorithms/tree/master/src/algorithms/graph/bellman-ford) - finding shortest path to all graph vertices
71-
* Detect Cycle
71+
* [Detect Cycle](https://github.com/trekhleb/javascript-algorithms/tree/master/src/algorithms/graph/detect-cycle)
7272
* Topological Sorting
7373
* Eulerian path, Eulerian circuit
7474
* Strongly Connected Component algorithm
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
# Detect Cycle in Graphs
2+
3+
In graph theory, a **cycle** is a path of edges and vertices
4+
wherein a vertex is reachable from itself. There are several
5+
different types of cycles, principally a **closed walk** and
6+
a **simple cycle**.
7+
8+
## Definitions
9+
10+
A **closed walk** consists of a sequence of vertices starting
11+
and ending at the same vertex, with each two consecutive vertices
12+
in the sequence adjacent to each other in the graph. In a directed graph,
13+
each edge must be traversed by the walk consistently with its direction:
14+
the edge must be oriented from the earlier of two consecutive vertices
15+
to the later of the two vertices in the sequence.
16+
The choice of starting vertex is not important: traversing the same cyclic
17+
sequence of edges from different starting vertices produces the same closed walk.
18+
19+
A **simple cycle may** be defined either as a closed walk with no repetitions of
20+
vertices and edges allowed, other than the repetition of the starting and ending
21+
vertex, or as the set of edges in such a walk. The two definitions are equivalent
22+
in directed graphs, where simple cycles are also called directed cycles: the cyclic
23+
sequence of vertices and edges in a walk is completely determined by the set of
24+
edges that it uses. In undirected graphs the set of edges of a cycle can be
25+
traversed by a walk in either of two directions, giving two possible directed cycles
26+
for every undirected cycle. A circuit can be a closed walk allowing repetitions of
27+
vertices but not edges; however, it can also be a simple cycle, so explicit
28+
definition is recommended when it is used.
29+
30+
## Example
31+
32+
![Cycles](https://upload.wikimedia.org/wikipedia/commons/e/e7/Graph_cycle.gif)
33+
34+
A graph with edges colored to illustrate **path** `H-A-B` (green), closed path or
35+
**walk with a repeated vertex** `B-D-E-F-D-C-B` (blue) and a **cycle with no repeated edge** or
36+
vertex `H-D-G-H` (red)
37+
38+
### Cycle in undirected graph
39+
40+
![Undirected Cycle](https://www.geeksforgeeks.org/wp-content/uploads/cycleGraph.png)
41+
42+
### Cycle in directed graph
43+
44+
![Directed Cycle](https://cdncontribute.geeksforgeeks.org/wp-content/uploads/cycle.png)
45+
46+
## References
47+
48+
General information:
49+
50+
- [Wikipedia](https://en.wikipedia.org/wiki/Cycle_(graph_theory))
51+
52+
Cycles in undirected graphs:
53+
54+
- [Detect Cycle in Undirected Graph on GeeksForGeeks](https://www.geeksforgeeks.org/detect-cycle-undirected-graph/)
55+
- [Detect Cycle in Undirected Graph Algorithm on YouTube](https://www.youtube.com/watch?v=n_t0a_8H8VY)
56+
57+
Cycles in directed graphs:
58+
59+
- [Detect Cycle in Directed Graph on GeeksForGeeks](https://www.geeksforgeeks.org/detect-cycle-in-a-graph/)
60+
- [Detect Cycle in Directed Graph Algorithm on YouTube](https://www.youtube.com/watch?v=rKQaZuoUR4M)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
import GraphVertex from '../../../../data-structures/graph/GraphVertex';
2+
import GraphEdge from '../../../../data-structures/graph/GraphEdge';
3+
import Graph from '../../../../data-structures/graph/Graph';
4+
import detectUndirectedCycle from '../detectUndirectedCycle';
5+
6+
describe('detectUndirectedCycle', () => {
7+
it('should detect undirected cycle', () => {
8+
const vertexA = new GraphVertex('A');
9+
const vertexB = new GraphVertex('B');
10+
const vertexC = new GraphVertex('C');
11+
const vertexD = new GraphVertex('D');
12+
const vertexE = new GraphVertex('E');
13+
const vertexF = new GraphVertex('F');
14+
15+
const edgeAF = new GraphEdge(vertexA, vertexF);
16+
const edgeAB = new GraphEdge(vertexA, vertexB);
17+
const edgeBE = new GraphEdge(vertexB, vertexE);
18+
const edgeBC = new GraphEdge(vertexB, vertexC);
19+
const edgeCD = new GraphEdge(vertexC, vertexD);
20+
const edgeDE = new GraphEdge(vertexD, vertexE);
21+
22+
const graph = new Graph();
23+
graph
24+
.addEdge(edgeAF)
25+
.addEdge(edgeAB)
26+
.addEdge(edgeBE)
27+
.addEdge(edgeBC)
28+
.addEdge(edgeCD);
29+
30+
expect(detectUndirectedCycle(graph)).toBeFalsy();
31+
32+
graph.addEdge(edgeDE);
33+
34+
expect(detectUndirectedCycle(graph)).toBeTruthy();
35+
});
36+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
import DisjointSet from '../../../data-structures/disjoint-set/DisjointSet';
2+
3+
/**
4+
* Detect cycle in undirected graph using disjoint sets.
5+
*
6+
* @param {Graph} graph
7+
*/
8+
9+
export default function detectUndirectedCycle(graph) {
10+
// Create initial singleton disjoint sets for each graph vertex.
11+
/** @param {GraphVertex} graphVertex */
12+
const keyExtractor = graphVertex => graphVertex.getKey();
13+
const disjointSet = new DisjointSet(keyExtractor);
14+
graph.getAllVertices().forEach(graphVertex => disjointSet.makeSet(graphVertex));
15+
16+
// Go trough all graph edges one by one and check if edge vertices are from the
17+
// different sets. In this case joint those sets together. Do this until you find
18+
// an edge where to edge vertices are already in one set. This means that current
19+
// edge will create a cycle.
20+
let cycleFound = false;
21+
/** @param {GraphEdge} graphEdge */
22+
graph.getAllEdges().forEach((graphEdge) => {
23+
if (disjointSet.inSameSet(graphEdge.startVertex, graphEdge.endVertex)) {
24+
// Cycle found.
25+
cycleFound = true;
26+
} else {
27+
disjointSet.union(graphEdge.startVertex, graphEdge.endVertex);
28+
}
29+
});
30+
31+
return cycleFound;
32+
}

src/data-structures/graph/Graph.js

+14-1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ export default class Graph {
44
*/
55
constructor(isDirected = false) {
66
this.vertices = {};
7+
this.edges = {};
78
this.isDirected = isDirected;
89
}
910

@@ -39,6 +40,13 @@ export default class Graph {
3940
return Object.values(this.vertices);
4041
}
4142

43+
/**
44+
* @return {GraphEdge[]}
45+
*/
46+
getAllEdges() {
47+
return Object.values(this.edges);
48+
}
49+
4250
/**
4351
* @param {GraphEdge} edge
4452
* @returns {Graph}
@@ -60,7 +68,12 @@ export default class Graph {
6068
endVertex = this.getVertexByKey(edge.endVertex.getKey());
6169
}
6270

63-
// @TODO: Check if edge has been already added.
71+
// Check if edge has been already added.
72+
if (this.edges[edge.getKey()]) {
73+
throw new Error('Edge has already been added before');
74+
} else {
75+
this.edges[edge.getKey()] = edge;
76+
}
6477

6578
// Add edge to the vertices.
6679
if (this.isDirected) {

src/data-structures/graph/GraphEdge.js

+10
Original file line numberDiff line numberDiff line change
@@ -9,4 +9,14 @@ export default class GraphEdge {
99
this.endVertex = endVertex;
1010
this.weight = weight;
1111
}
12+
13+
/**
14+
* @return {string}
15+
*/
16+
getKey() {
17+
const startVertexKey = this.startVertex.getKey();
18+
const endVertexKey = this.endVertex.getKey();
19+
20+
return `${startVertexKey}_${endVertexKey}`;
21+
}
1222
}

src/data-structures/graph/__test__/Graph.test.js

+38
Original file line numberDiff line numberDiff line change
@@ -136,4 +136,42 @@ describe('Graph', () => {
136136
expect(neighbors[0]).toEqual(vertexB);
137137
expect(neighbors[1]).toEqual(vertexC);
138138
});
139+
140+
it('should throw an error when trying to add edge twice', () => {
141+
function addSameEdgeTwice() {
142+
const graph = new Graph(true);
143+
144+
const vertexA = new GraphVertex('A');
145+
const vertexB = new GraphVertex('B');
146+
147+
const edgeAB = new GraphEdge(vertexA, vertexB);
148+
149+
graph
150+
.addEdge(edgeAB)
151+
.addEdge(edgeAB);
152+
}
153+
154+
expect(addSameEdgeTwice).toThrow();
155+
});
156+
157+
it('should return the list of all added edges', () => {
158+
const graph = new Graph(true);
159+
160+
const vertexA = new GraphVertex('A');
161+
const vertexB = new GraphVertex('B');
162+
const vertexC = new GraphVertex('C');
163+
164+
const edgeAB = new GraphEdge(vertexA, vertexB);
165+
const edgeBC = new GraphEdge(vertexB, vertexC);
166+
167+
graph
168+
.addEdge(edgeAB)
169+
.addEdge(edgeBC);
170+
171+
const edges = graph.getAllEdges();
172+
173+
expect(edges.length).toBe(2);
174+
expect(edges[0]).toEqual(edgeAB);
175+
expect(edges[1]).toEqual(edgeBC);
176+
});
139177
});

src/data-structures/graph/__test__/GraphEdge.test.js

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ describe('GraphEdge', () => {
77
const endVertex = new GraphVertex('B');
88
const edge = new GraphEdge(startVertex, endVertex);
99

10+
expect(edge.getKey()).toBe('A_B');
1011
expect(edge.startVertex).toEqual(startVertex);
1112
expect(edge.endVertex).toEqual(endVertex);
1213
expect(edge.weight).toEqual(1);

0 commit comments

Comments
 (0)