-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy path04) Kosaraju SCC.py
130 lines (84 loc) · 2.7 KB
/
04) Kosaraju SCC.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
'''
Programming assignment 4:
implement the Kosaraju's
algorithm for computing
strongly connected components (SCC's)
in the given directed graph.
Find 5 largest SCC's.
@author: Mikhail Dubov
'''
import sys
import threading
def readDirectedGraph(filename):
f = open(filename)
adjlist = []
adjlist_reversed = []
line = f.readline()
while line != '':
num1, num2 = line.split()
v_from = int(num1)
v_to = int(num2)
max_v = max(v_from, v_to)
while len(adjlist) < max_v:
adjlist.append([])
while len(adjlist_reversed) < max_v:
adjlist_reversed.append([])
adjlist[v_from-1].append(v_to-1)
adjlist_reversed[v_to-1].append(v_from-1)
line = f.readline()
return adjlist, adjlist_reversed
t = 0
s = None
explored = None
leader = None
scc_size = 0
sorted_by_finish_time = None
def DFS_Loop_1(graph_rev, n):
global t, explored, sorted_by_finish_time
t = 0
explored = [False]*n
sorted_by_finish_time = [None]*n
for i in reversed(range(n)):
if not explored[i]:
DFS_1(graph_rev, i)
def DFS_1(graph_rev, i):
global t, explored
explored[i] = True
for v in graph_rev[i]:
if not explored[v]:
DFS_1(graph_rev, v)
sorted_by_finish_time[t] = i
t += 1
def DFS_Loop_2(graph):
global scc_size, explored, sorted_by_finish_time
explored = [False]*len(graph)
res = []
for i in reversed(range(len(graph))):
if not explored[sorted_by_finish_time[i]]:
scc_size = 0
# Here we collect all the members
# of the next SCC using DFS.
# scc_size is incremented inside DFS.
DFS_2(graph, sorted_by_finish_time[i])
res.append(scc_size)
return res
def DFS_2(graph, i):
global explored, scc_size
explored[i] = True
for v in graph[i]:
if not explored[v]:
DFS_2(graph, v)
scc_size += 1
def kosarajuSCCsizes(graph, graph_rev):
DFS_Loop_1(graph_rev, len(graph))
res = DFS_Loop_2(graph)
return res
def main():
graph, graph_rev = readDirectedGraph('SCC.txt')
res = kosarajuSCCsizes(graph, graph_rev)
print(','.join(map(lambda x: str(x), sorted(res)[::-1][:5])))
if __name__ == '__main__':
threading.stack_size(67108864) # 64MB stack
sys.setrecursionlimit(2 ** 20) # approx 1 million recursions
thread = threading.Thread(target = main) # instantiate thread object
thread.start() # run program at target