-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstring_multimatching_aho_corasick.py
44 lines (42 loc) · 1.46 KB
/
string_multimatching_aho_corasick.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
# Using Aho-Corasick
from collections import deque
def smm(s, pp):
A = 96
a = [[[-1]*A, 0, -1, []]]
for w in range(len(pp)):
n = 0
for i in range(len(pp[w])):
idx = ord(pp[w][i])-32
if a[n][0][idx] == -1: a[n][0][idx] = len(a); a.append([[-1]*A, 0, -1, []])
n = a[n][0][idx]
a[n][3].append(w); n = 0
q = deque()
for k in range(A):
if a[0][0][k] == -1: a[0][0][k] = 0
elif a[0][0][k] > 0: a[a[0][0][k]][1] = 0; q.append(a[0][0][k])
while q:
r = q.popleft()
for k in range(A):
arck = a[r][0][k]
if arck != -1:
q.append(arck)
v = a[r][1]
while a[v][0][k] == -1: v = a[v][1]
a[arck][1] = a[arck][2] = a[v][0][k]
while a[arck][2] != -1 and not a[a[arck][2]][3]: a[arck][2] = a[a[arck][2]][2]
matches = [[] for _ in range(len(pp))]
state = ss = 0
for i in range(len(s)):
idx = ord(s[i])-32
while a[ss][0][idx] == -1: ss = a[ss][1]
a[state][0][idx] = a[ss][0][idx]; state = a[state][0][idx]; ss = state
while ss != -1:
for w in a[ss][3]: matches[w].append(i+1-len(pp[w]))
ss = a[ss][2]
ss = state
n = -1
return matches
if __name__ == '__main__':
s = 'banana boy likes apple boys'
pp = ['a', 'na', 'boy', 'p', 'orange']
print(smm(s, pp))