Skip to content

Commit cf27c00

Browse files
committed
Fix topological sort bug, improve explanation
When concatenating the results from the individual depth-first searches, the subsequent results need to be prepended, not appended, to the final list.
1 parent 05ef99b commit cf27c00

29 files changed

+245
-161
lines changed

README.markdown

+1-1
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@ Special-purpose sorts:
7575
- Bucket Sort
7676
- Counting Sort
7777
- Radix Sort
78-
- [Topological Sort](Topological Sort/) :construction:
78+
- [Topological Sort](Topological Sort/)
7979

8080
Bad sorting algorithms (don't use these!):
8181

Topological Sort/Graph.swift

+19
Original file line numberDiff line numberDiff line change
@@ -30,3 +30,22 @@ public class Graph: CustomStringConvertible {
3030
return nil
3131
}
3232
}
33+
34+
extension Graph {
35+
typealias InDegree = Int
36+
37+
func calculateInDegreeOfNodes() -> [Node : InDegree] {
38+
var inDegrees = [Node : InDegree]()
39+
40+
for (node, _) in adjacencyLists {
41+
inDegrees[node] = 0
42+
}
43+
44+
for (_, adjacencyList) in adjacencyLists {
45+
for nodeInList in adjacencyList {
46+
inDegrees[nodeInList] = (inDegrees[nodeInList] ?? 0) + 1
47+
}
48+
}
49+
return inDegrees
50+
}
51+
}
2.08 KB
Binary file not shown.
25 KB
Loading
2.1 KB
Binary file not shown.

Topological Sort/Images/Example.png

9.09 KB
Loading

Topological Sort/Images/Graph.graffle

2.17 KB
Binary file not shown.

Topological Sort/Images/Graph.png

7.94 KB
Loading
2 Bytes
Binary file not shown.
1.34 KB
Loading
2.24 KB
Binary file not shown.
7.16 KB
Loading
-248 Bytes
Binary file not shown.
-10.6 KB
Loading
-183 KB
Binary file not shown.
-111 KB
Binary file not shown.

Topological Sort/README.markdown

+86-30
Original file line numberDiff line numberDiff line change
@@ -1,76 +1,132 @@
11
# Topological Sort
22

3-
Topological sort is an algorithm aimed at ordering a directed graph such that for each directed edge *uv* from vertex *u* to vertex *v*, *u* comes before *v*.
3+
Topological sort is an algorithm that orders a directed graph such that for each directed edge *u→v*, vertex *u* comes before vertex *v*.
44

5-
In other words, a topological sort places the vertices of a directed acyclic graph on a line so that all directed edges go from left to right.
5+
In other words, a topological sort places the vertices of a [directed acyclic graph](../Graph/) on a line so that all directed edges go from left to right.
66

7-
The graph in the following example has two possible topological sorts:
7+
Consider the graph in the following example:
8+
9+
![Example](Images/Graph.png)
10+
11+
This graph has two possible topological sorts:
812

913
![Example](Images/TopologicalSort.png)
1014

15+
The topological orderings are **S, V, W, T, X** and **S, W, V, T, X**. Notice how the arrows all go from left to right.
16+
17+
The following is not a valid topological sort for this graph, since **X** and **T** cannot happen before **V**:
18+
19+
![Example](Images/InvalidSort.png)
20+
1121
## Where is this used?
1222

13-
Let's consider that you want to learn all the algorithms present in the Swift Algorithm Club. This might seem daunting at first but we can use topological sort to get things organized.
23+
Let's consider that you want to learn all the algorithms and data structures from the Swift Algorithm Club. This might seem daunting at first but we can use topological sort to get things organized.
1424

15-
For example, to learn the [depth-first search](../Depth-First Search/) algorithm you need to know how a [graph](../Graph/) is represented. Similarly, to understand how to calculate the length of a [binary tree](../Binary Tree/) you need to know details of how a [tree is traversed](../Tree/).
25+
Since you're learning about topological sort, let's take this topic as an example. What else do you need to learn first before you can fully understand topological sort? Well, topological sort uses [depth-first search](../Depth-First Search/) as well as a [stack](../Stack/). But before you can learn about the depth-first search algorithm, you need to know what a [graph](../Graph/) is, and it helps to know what a [tree](../Tree/) is. In turn, graphs and trees use the idea of linking objects together, so you may need to read up on that first. And so on...
1626

17-
If we were to represent these objectives in the form of a graph it would be as follows:
27+
If we were to represent these objectives in the form of a graph it would look as follows:
1828

19-
![Example](Images/algorithm_example.png)
29+
![Example](Images/Algorithms.png)
2030

21-
If we consider each algorithm to be a node in the graph you'll see dependancies among them, i.e. to learn something you might have to know something else first. This is exactly what topological sort is used for -- it will sort things out such that you know what to learn first.
31+
If we consider each algorithm to be a vertex in the graph you can clearly see the dependencies between them. To learn something you might have to know something else first. This is exactly what topological sort is used for -- it will sort things out such that you know what to do first.
2232

2333
## How does it work?
2434

25-
**Step 1: Find all nodes that have in-degree of 0**
35+
**Step 1: Find all vertices that have in-degree of 0**
36+
37+
The *in-degree* of a vertex is the number of edges pointing at that vertex. Vertices with no incoming edges have an in-degree of 0. These vertices are the starting points for the topological sort.
38+
39+
In the context of the previous example, these vertices represent algorithms that don't have any prerequisites; you don't need to learn anything else first, hence the sort starts with them.
2640

27-
The *in-degree* of a node is the number of incoming edges to that node. Nodes that have no incoming edges have an in-degree of 0. These nodes are the starting points for the sort.
41+
**Step 2: Traverse the graph with depth-first search**
2842

29-
If you think about it in the context of the previous example, these nodes represent algorithms that don't have any prerequisites; you don't need to learn anything else first, hence the sort starts with them.
43+
Depth-first search is an algorithm that starts traversing the graph from a certain vertex and explores as far as possible along each branch before backtracking. To find out more about depth-first search, please take a look at the [detailed explanation](../Depth-First%20Search/).
3044

31-
**Step 2: Depth-first search for traversal**
45+
We perform a depth-first search on each vertex with in-degree 0. This tells us which vertices are connected to each of these starting vertices.
3246

33-
Depth-first search is an algorithm that is used to traverse a graph. This algorithm traverses all the child nodes recursively and uses backtracking.
47+
**Step 3: Remember all visited vertices**
3448

35-
To find out more about depth-first search, please take a look at the [detailed explanation](../Depth-First%20Search/).
49+
As we perform the depth-first search, we maintain a list of all the vertices that have been visited. This is to avoid visiting the same vertex twice.
3650

37-
**Step 3: Remember all visited nodes**
51+
**Step 4: Put it all together**
3852

39-
The last step of the sort is to maintain a list of all the nodes that have been visited. This is to avoid visiting the same node twice.
53+
The last step of the sort is to combine the results of the different depth-first searches and put the vertices in a sorted list.
4054

4155
## Example
4256

4357
Consider the following graph:
4458

45-
![Graph Example](Images/graph_example.png)
59+
![Graph Example](Images/Example.png)
4660

47-
**Step 1:** The nodes with 0 in-degree are: **5, 7, 3**
61+
**Step 1:** The vertices with 0 in-degree are: **3, 7, 5**. These are our starting vertices.
4862

49-
**Step 2:** Depth-first search for each starting node, without remembering nodes that have already been visited:
63+
**Step 2:** Perform depth-first search for each starting vertex, without remembering vertices that have already been visited:
5064

5165
```
52-
Node 3: 3, 8, 9, 10
53-
Node 7: 7, 11, 2, 8, 9
54-
Node 5: 5, 11, 2, 9, 10
66+
Vertex 3: 3, 8, 9, 10
67+
Vertex 7: 7, 11, 2, 8, 9
68+
Vertex 5: 5, 11, 2, 9, 10
5569
```
5670

57-
**Step 3:** Remember nodes already visited in each DFS:
71+
**Step 3:** Filter out the vertices already visited in each previous search:
5872

5973
```
60-
Node 3: 3, 8, 9, 10
61-
Node 7: 7, 11, 2
62-
Node 5: 5
74+
Vertex 3: 3, 8, 9, 10
75+
Vertex 7: 7, 11, 2
76+
Vertex 5: 5
6377
```
6478

65-
The final sorted order is a concatenation of the above three traversals: **3, 8, 9, 10, 7, 11, 2, 5**
79+
**Step 4:** Combine the results of these three depth-first searches. The final sorted order is **5, 7, 11, 2, 3, 8, 9, 10**. (Important: we need to add the results of each subsequent search to the *front* of the list.)
6680

6781
The result of the topological sort looks like this:
6882

6983
![Result of the sort](Images/GraphResult.png)
7084

71-
TODO: I don't think this is correct! There should be no arrows going from right to left! (A valid topological order would be 3, 7, 5, 10, 8, 11, 9, 2 -- or 3, 7, 5, 8, 11, 2, 9, 10.)
85+
> **Note:** This is not the only possible topological sort for this graph. For example, other valid solutions are **3, 7, 5, 10, 8, 11, 9, 2** and **3, 7, 5, 8, 11, 2, 9, 10**. Any order where all the arrows are going from left to right will do.
86+
87+
## The code
88+
89+
Here is how you could implement topological sort in Swift (see also [TopologicalSort1.swift](TopologicalSort1.swift)):
90+
91+
```swift
92+
extension Graph {
93+
public func topologicalSort() -> [Node] {
94+
// 1
95+
let startNodes = calculateInDegreeOfNodes().filter({ _, indegree in
96+
return indegree == 0
97+
}).map({ node, indegree in
98+
return node
99+
})
100+
101+
// 2
102+
var visited = [Node : Bool]()
103+
for (node, _) in adjacencyLists {
104+
visited[node] = false
105+
}
106+
107+
// 3
108+
var result = [Node]()
109+
for startNode in startNodes {
110+
result = depthFirstSearch(startNode, visited: &visited) + result
111+
}
112+
113+
// 4
114+
return result
115+
}
116+
}
117+
```
118+
119+
Some remarks:
120+
121+
1. Find the in-degree of each vertex and put all the vertices with in-degree 0 in the `startNodes` array. In this graph implementation, vertices are called "nodes". Both terms are used interchangeably by people who write graph code.
122+
123+
2. The `visited` array keeps track of whether we've already seen a vertex during the depth-first search. Initially, we set all elements to `false`.
124+
125+
3. For each of the vertices in the `startNodes` array, perform a depth-first search. This returns an array of `Node` objects. We prepend that array to our own `result` array.
126+
127+
4. The `result` array contains all the vertices in topologically sorted order.
72128

73-
## Alternative algorithm
129+
## Alternative method (Kahn's algorithm)
74130

75131
Even though depth-first search is the typical way to perform a topological sort, there is another algorithm that also does the job.
76132

@@ -82,6 +138,6 @@ Even though depth-first search is the typical way to perform a topological sort,
82138

83139
This is an **O(n + m)** algorithm where **n** is the number of vertices and **m** is the number of edges. You can see the implementation in [TopologicalSort2.swift](TopologicalSort2.swift).
84140

85-
I first read about this algorithm in the Algorithm Alley column in Dr. Dobb's Magazine from May 1993.
141+
Source: I first read about this alternative algorithm in the Algorithm Alley column in Dr. Dobb's Magazine from May 1993.
86142

87143
*Written for Swift Algorithm Club by Ali Hafizji and Matthijs Hollemans*

Topological Sort/Stack.swift

-27
This file was deleted.

Topological Sort/Tests/Tests.xcodeproj/project.pbxproj

+4-4
Original file line numberDiff line numberDiff line change
@@ -9,19 +9,19 @@
99
/* Begin PBXBuildFile section */
1010
7B3471BB1C8F50CF008381CD /* TopologicalSortTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B3471BA1C8F50CF008381CD /* TopologicalSortTests.swift */; };
1111
7B3471BF1C8F50DD008381CD /* Graph.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B3471BC1C8F50DD008381CD /* Graph.swift */; };
12-
7B3471C01C8F50DD008381CD /* Stack.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B3471BD1C8F50DD008381CD /* Stack.swift */; };
1312
7B3471C11C8F50DD008381CD /* TopologicalSort1.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B3471BE1C8F50DD008381CD /* TopologicalSort1.swift */; };
1413
7B3471C31C8F5348008381CD /* TopologicalSort2.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B3471C21C8F5348008381CD /* TopologicalSort2.swift */; };
14+
7BE143491C9C6A93001BC747 /* TopologicalSort3.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BE143481C9C6A93001BC747 /* TopologicalSort3.swift */; };
1515
/* End PBXBuildFile section */
1616

1717
/* Begin PBXFileReference section */
1818
7B2BBC801C779D720067B71D /* Tests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = Tests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
1919
7B2BBC941C779E7B0067B71D /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = SOURCE_ROOT; };
2020
7B3471BA1C8F50CF008381CD /* TopologicalSortTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = TopologicalSortTests.swift; sourceTree = SOURCE_ROOT; };
2121
7B3471BC1C8F50DD008381CD /* Graph.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Graph.swift; path = ../Graph.swift; sourceTree = SOURCE_ROOT; };
22-
7B3471BD1C8F50DD008381CD /* Stack.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Stack.swift; path = ../Stack.swift; sourceTree = SOURCE_ROOT; };
2322
7B3471BE1C8F50DD008381CD /* TopologicalSort1.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = TopologicalSort1.swift; path = ../TopologicalSort1.swift; sourceTree = SOURCE_ROOT; };
2423
7B3471C21C8F5348008381CD /* TopologicalSort2.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = TopologicalSort2.swift; path = ../TopologicalSort2.swift; sourceTree = SOURCE_ROOT; };
24+
7BE143481C9C6A93001BC747 /* TopologicalSort3.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = TopologicalSort3.swift; path = ../TopologicalSort3.swift; sourceTree = SOURCE_ROOT; };
2525
/* End PBXFileReference section */
2626

2727
/* Begin PBXFrameworksBuildPhase section */
@@ -55,9 +55,9 @@
5555
isa = PBXGroup;
5656
children = (
5757
7B3471BC1C8F50DD008381CD /* Graph.swift */,
58-
7B3471BD1C8F50DD008381CD /* Stack.swift */,
5958
7B3471BE1C8F50DD008381CD /* TopologicalSort1.swift */,
6059
7B3471C21C8F5348008381CD /* TopologicalSort2.swift */,
60+
7BE143481C9C6A93001BC747 /* TopologicalSort3.swift */,
6161
7B3471BA1C8F50CF008381CD /* TopologicalSortTests.swift */,
6262
7B2BBC941C779E7B0067B71D /* Info.plist */,
6363
);
@@ -135,9 +135,9 @@
135135
files = (
136136
7B3471C31C8F5348008381CD /* TopologicalSort2.swift in Sources */,
137137
7B3471BF1C8F50DD008381CD /* Graph.swift in Sources */,
138-
7B3471C01C8F50DD008381CD /* Stack.swift in Sources */,
139138
7B3471BB1C8F50CF008381CD /* TopologicalSortTests.swift in Sources */,
140139
7B3471C11C8F50DD008381CD /* TopologicalSort1.swift in Sources */,
140+
7BE143491C9C6A93001BC747 /* TopologicalSort3.swift in Sources */,
141141
);
142142
runOnlyForDeploymentPostprocessing = 0;
143143
};

Topological Sort/Tests/TopologicalSortTests.swift

+24-17
Original file line numberDiff line numberDiff line change
@@ -40,44 +40,51 @@ class TopologicalSort: XCTestCase {
4040
graph.addEdge(fromNode: node11, toNode: node10)
4141
graph.addEdge(fromNode: node8, toNode: node9)
4242

43-
XCTAssertEqual(graph.topologicalSort(), ["3", "8", "9", "10", "7", "11", "2", "5"])
44-
XCTAssertEqual(graph.topologicalSort2(), ["3", "7", "5", "8", "11", "2", "9", "10"])
43+
XCTAssertEqual(graph.topologicalSort(), ["5", "7", "11", "2", "3", "8", "9", "10"])
44+
XCTAssertEqual(graph.topologicalSortKahn(), ["3", "7", "5", "8", "11", "2", "9", "10"])
45+
XCTAssertEqual(graph.topologicalSortAlternative(), ["5", "7", "3", "8", "11", "10", "9", "2"])
4546
}
4647

4748
func testTopologicalSortEdgeLists() {
4849
let p1 = ["A B", "A C", "B C", "B D", "C E", "C F", "E D", "F E", "G A", "G F"]
49-
let a1 = ["G", "A", "B", "C", "F", "E", "D"] // TODO
50-
let s1 = ["G", "A", "B", "C", "F", "E", "D"]
50+
let s1 = ["G", "A", "B", "C", "E", "D", "F"]
51+
let a1 = ["G", "A", "B", "C", "F", "E", "D"]
52+
let k1 = ["G", "A", "B", "C", "F", "E", "D"]
5153

5254
let p2 = ["B C", "C D", "C G", "B F", "D G", "G E", "F G", "F G"]
53-
let a2 = ["B", "C", "F", "D", "G", "E"] // TODO
54-
let s2 = ["B", "C", "F", "D", "G", "E"]
55+
let s2 = ["B", "C", "D", "G", "E", "F"]
56+
let a2 = ["B", "C", "F", "D", "G", "E"]
57+
let k2 = ["B", "F", "C", "D", "G", "E"]
5558

5659
let p3 = ["S V", "S W", "V T", "W T"]
57-
let a3 = ["S", "V", "W", "T"] // TODO
5860
let s3 = ["S", "V", "W", "T"]
61+
let a3 = ["S", "V", "W", "T"]
62+
let k3 = ["S", "V", "W", "T"]
5963

6064
let p4 = ["5 11", "7 11", "7 8", "3 8", "3 10", "11 2", "11 9", "11 10", "8 9"]
61-
let a4 = ["3", "8", "9", "10", "7", "11", "2", "5"]
62-
let s4 = ["3", "7", "5", "8", "11", "2", "9", "10"]
65+
let s4 = ["5", "7", "11", "2", "3", "8", "9", "10"]
66+
let a4 = ["3", "7", "5", "8", "11", "2", "9", "10"]
67+
let k4 = ["5", "7", "3", "8", "11", "10", "9", "2"]
6368

6469
let data = [
65-
(p1, a1, s1),
66-
(p2, a2, s2),
67-
(p3, a3, s3),
68-
(p4, a4, s4),
70+
(p1, s1, a1, k1),
71+
(p2, s2, a2, k2),
72+
(p3, s3, a3, k3),
73+
(p4, s4, a4, k4),
6974
]
7075

7176
for d in data {
7277
let graph = Graph()
7378
graph.loadEdgeList(d.0)
7479

75-
// TODO: this fails the tests
76-
//let sorted1 = graph.topologicalSort()
77-
//XCTAssertEqual(sorted1, d.1)
80+
let sorted1 = graph.topologicalSort()
81+
XCTAssertEqual(sorted1, d.1)
7882

79-
let sorted2 = graph.topologicalSort2()
83+
let sorted2 = graph.topologicalSortKahn()
8084
XCTAssertEqual(sorted2, d.2)
85+
86+
let sorted3 = graph.topologicalSortAlternative()
87+
XCTAssertEqual(sorted3, d.3)
8188
}
8289
}
8390
}

Topological Sort/Topological Sort.playground/Contents.swift

+5-2
Original file line numberDiff line numberDiff line change
@@ -28,5 +28,8 @@ graph.addEdge(fromNode: node8, toNode: node9)
2828
// using depth-first search
2929
graph.topologicalSort()
3030

31-
// alternative method
32-
graph.topologicalSort2()
31+
// also using depth-first search
32+
graph.topologicalSortAlternative()
33+
34+
// Kahn's algorithm
35+
graph.topologicalSortKahn()

Topological Sort/Topological Sort.playground/Sources/Graph.swift

+19
Original file line numberDiff line numberDiff line change
@@ -30,3 +30,22 @@ public class Graph: CustomStringConvertible {
3030
return nil
3131
}
3232
}
33+
34+
extension Graph {
35+
typealias InDegree = Int
36+
37+
func calculateInDegreeOfNodes() -> [Node : InDegree] {
38+
var inDegrees = [Node : InDegree]()
39+
40+
for (node, _) in adjacencyLists {
41+
inDegrees[node] = 0
42+
}
43+
44+
for (_, adjacencyList) in adjacencyLists {
45+
for nodeInList in adjacencyList {
46+
inDegrees[nodeInList] = (inDegrees[nodeInList] ?? 0) + 1
47+
}
48+
}
49+
return inDegrees
50+
}
51+
}

0 commit comments

Comments
 (0)