-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathtop-sort.py
82 lines (60 loc) · 1.89 KB
/
top-sort.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
# !/usr/bin/env python
# -*- coding: utf-8 -*-
"""
Topological sorting
"""
__author__ = "prakashsellathurai"
__copyright__ = "Copyright 2021"
__version__ = "1.0.1"
__email__ = "prakashsellathurai@gmail.com"
from random import randrange
from collections import defaultdict
def naive_topsort(G, S=None):
if S is None:
S = set(G)
if len(S) == 1:
return list(S)
v = S.pop()
seq = naive_topsort(G, S)
min_i = 0
for i, u in enumerate(seq):
if v in G[u]:
min_i = i + 1
seq.insert(min_i, v)
return seq
def topsort(G):
count = {u: 0 for u in G} # The in-degree for each node
for u in G:
for v in G[u]:
count[v] += 1 # Count every in-edge
Q = [u for u in G if count[u] == 0] # Valid initial nodes
S = [] # The result
while Q: # While we have start nodes...
u = Q.pop() # Pick one
S.append(u) # Use it as first of the rest
for v in G[u]:
count[v] -= 1 # "Uncount" its out-edges
if count[v] == 0: # New valid start nodes?
Q.append(v) # Deal with them next
return S
class randRangesparse:
"""docstring for randRangesparse."""
def __init__(self, n):
super(randRangesparse, self).__init__()
self.n = n
self.weights = self.regenerate_weights()
def regenerate_weights(self):
return {i: self.n - i for i in range(self.n)}
def get_random(self):
coin = randrange(self.n)
while self.weights[coin] != 0:
if self.weights[coin] != 0:
self.weights[coin] -= 1
return coin
coin = randrange(self.n - 1)
self.weights = self.regenerate_weights()
return self.get_random()
if __name__ == "__main__":
rng = randRangesparse(10)
G = [[rng.get_random() for _ in range(10)] for _ in range(10)]
print(naive_topsort(G))