-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsuffix_array_long.py
84 lines (60 loc) · 1.99 KB
/
suffix_array_long.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
# python3
import sys
def sort_characters(text):
order = [0] * len(text)
chars={'$':0,'A':1,'C':2,'G':3,'T':4}
count = [0]*5
for i in text:
count[chars[i]] += 1
for i in range(1, len(count)):
count[i] += count[i - 1]
for i, c in reversed(list(enumerate(text))):
count[chars[c]] -= 1
order[count[chars[c]]] = i
return order
def compute_char_classes(text, order):
classs = [0] * len(text)
for i in range(1, len(text)):
if text[order[i]] != text[order[i - 1]]:
classs[order[i]] = classs[order[i - 1]] + 1
else:
classs[order[i]] = classs[order[i - 1]]
return classs
def sort_doubled(text, l, order, classs):
len_text = len(text)
count = [0] * len_text
new_order = [0] * len_text
for i in range(len_text):
count[classs[i]] += 1
for j in range(1, len_text):
count[j] += count[j - 1]
for i in range(len_text - 1, -1, -1):
start = (order[i] - l + len_text) % len_text
cl = classs[start]
count[cl] -= 1
new_order[count[cl]] = start
return new_order
def update_classes(new_order, classs, l):
n = len(new_order)
new_classs = [0] * n
for i in range(1, n):
cur, prev = new_order[i], new_order[i - 1]
mid, mid_prev = cur + l, (prev + l) % n
if classs[cur] != classs[prev] or classs[mid] != classs[mid_prev]:
new_classs[cur] = new_classs[prev] + 1
else:
new_classs[cur] = new_classs[prev]
return new_classs
def build_suffix_array(text):
order = sort_characters(text)
classs = compute_char_classes(text, order)
length = 1
len_text = len(text)
while length < len_text:
order = sort_doubled(text, length, order, classs)
classs = update_classes(order, classs, length)
length = length * 2
return order
if __name__ == '__main__':
text = sys.stdin.readline().strip()
print(" ".join(map(str, build_suffix_array(text))))