forked from indy256/codelibrary
-
Notifications
You must be signed in to change notification settings - Fork 0
/
aho-corasick.cpp
59 lines (51 loc) · 1.29 KB
/
aho-corasick.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
#include <bits/stdc++.h>
using namespace std;
constexpr int ALPHABET_SIZE = 26;
constexpr int MAX_STATES = 200'000;
int transitions[MAX_STATES][ALPHABET_SIZE];
int sufflink[MAX_STATES];
int escape[MAX_STATES];
int states = 1;
int addString(const string &s) {
int v = 0;
for (char c : s) {
c -= 'a';
if (transitions[v][c] == 0) {
transitions[v][c] = states++;
}
v = transitions[v][c];
}
escape[v] = v;
return v;
}
void buildLinks() {
vector<int> q(MAX_STATES);
for (int s = 0, t = 1; s < t;) {
int v = q[s++];
int u = sufflink[v];
if (escape[v] == 0) {
escape[v] = escape[u];
}
for (int c = 0; c < ALPHABET_SIZE; c++) {
if (transitions[v][c] != 0) {
q[t++] = transitions[v][c];
sufflink[transitions[v][c]] = v != 0 ? transitions[u][c] : 0;
} else {
transitions[v][c] = transitions[u][c];
}
}
}
}
int main() {
addString("a");
addString("aa");
addString("abaaa");
buildLinks();
string s = "abaa";
int state = 0;
for (size_t i = 0; i < s.length(); i++) {
state = transitions[state][s[i] - 'a'];
if (escape[state] != 0)
cout << i << endl;
}
}