-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathze_suffarray.cpp
104 lines (83 loc) · 1.69 KB
/
ze_suffarray.cpp
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
#include <fstream>
#include <iostream>
#include <string>
#include <memory.h>
using namespace std;
string text;
int n;
int const maxLength = 200000;
int const alphabetSize = 256;
int p[maxLength];
int c[maxLength];
//int cnt[alphabetSize];
int cnt[maxLength];
// temporary arrays
int pn[maxLength];
int cn[maxLength];
int main() {
ifstream in("suffarray.in");
getline(in, text);
in.close();
text += '\0'; // adding just for cycle computation
n = text.length();
// first step sort
for (int i = 0; i < n; i++) {
cnt[(unsigned) text[i]]++;
}
for (int i = 1; i < alphabetSize; i++) {
cnt[i] += cnt[i - 1];
}
for (int i = 0; i < n; i++) {
cnt[(unsigned) text[i]]--;
p[cnt[(unsigned) text[i]]] = i;
}
c[p[0]] = 0;
int classes = 1;
for (int i = 1; i < n; i++) {
if (text[p[i]] != text[p[i - 1]]) {
classes++;
}
c[p[i]] = classes - 1;
}
// main part
for (int step = 0; (1 << step) <= n; step++) {
for (int i = 0; i < n; i++) {
pn[i] = p[i] - (1 << step);
if (pn[i] < 0) {
pn[i] += n;
}
}
memset(cnt, 0, classes * sizeof(int));
for (int i = 0; i < n; i++) {
cnt[c[pn[i]]]++;
}
for (int i = 1; i < classes; i++) {
cnt[i] += cnt[i - 1];
}
for (int i = n - 1; i >= 0; i--) {
cnt[c[pn[i]]]--;
p[cnt[c[pn[i]]]] = pn[i];
}
cn[p[0]] = 0;
classes = 1;
for (int i = 1; i < n; i++) {
int mid1 = (p[i] + (1 << step)) % n;
int mid2 = (p[i - 1] + (1 << step)) % n;
if (
c[p[i]] != c[p[i - 1]]
||
c[mid1] != c[mid2]
) {
classes++;
}
cn[p[i]] = classes - 1;
}
memcpy(c, cn, n * sizeof(int));
}
ofstream out("suffarray.out");
for (int i = 1; i < n; i++) {
out << p[i] + 1 << " ";
}
out.close();
return 0;
}