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
|
def SA(A):
def SA_IS(A):
def equal(p1,p2):
e1,e2 = LMS.find('*',p1+1),LMS.find('*',p2+1)
return A[p1:e1+1]==A[p2:e2+1]
def IS(stars):
sa = [n]+[-1]*n
tails = list(accumulate(bucket))
for i in stars[::-1]:
sa[tails[A[i]]] = i
tails[A[i]] -= 1
heads = list(accumulate([1]+bucket[:-1]))
for i in range(n+1):
j = sa[i]-1
if j>=0 and types[j]=='L':
sa[heads[A[j]]] = j
heads[A[j]] += 1
tails = list(accumulate(bucket))
for i in range(n,-1,-1):
j = sa[i]-1
if j >= 0 and types[j] == 'S':
sa[tails[A[j]]] = j
tails[A[j]] -= 1
return sa[1:]
n = len(A)
types = list('0'*(n-1)+'LS')
for i in range(n-2,-1,-1):
types[i] = 'S' if A[i]<A[i+1] else 'L' if A[i]>A[i+1] else types[i+1]
LMS = ''.join('*' if types[i-1:i+1]==['L','S'] else ' ' for i in range(n+1))
ct = Counter(A)
bucket = [ct[x] for x in range(max(ct)+1)]
stars = [i for i in range(n) if LMS[i]=='*']
sa = IS(stars)
d, cnt, prev = {}, 0, -1
for pos in sa:
if LMS[pos]=='*':
cnt += prev<0 or not equal(prev,pos)
d[pos] = cnt
prev = pos
B = [d[pos] for pos in stars]
d1 = {x-1:i for i,x in enumerate(B)}
sa1 = [d1[x] for x in range(cnt)] if cnt==len(B) else SA_IS(B)
return IS([stars[pos] for pos in sa1])
def SA_h(A,sa):
n = len(A)
rk, height, h = [0]*n,[0]*n,0
for i in range(n):
rk[sa[i]] = i
for i in range(n):
h = max(0, h-1)
j = sa[rk[i]-1] if rk[i] else n
while max(i,j)+h<n and A[i+h]==A[j+h]:
h += 1
height[rk[i]] = h
return rk, height
sa = SA_IS(A) # sa[i]:第i小的后缀编号
rk, height = SA_h(A,sa) # rk[i]:后缀i的排名
return sa,rk,height # height[i]:lcp(sa[i],sa[i-1])
class ST:
def __init__(self,A):
f = A[:]
self.st = st = [f]
j, N = 1, len(f)
while 2*j<=N:
f = [min(f[i],f[i+j]) for i in range(N-2*j+1)]
st.append(f)
j <<= 1
def query(self,l,r):
j = (r-l+1).bit_length()-1
return min(self.st[j][l],self.st[j][r-(1<<j)+1])
class Solution:
def longestDecomposition(self, text: str) -> int:
A = [ord(c)-ord('a') for c in text]
sa,rk,height = SA(A)
st = ST(height)
n = len(A)
res,i = 0,0
for j in range(n//2):
l,r = sorted([rk[i],rk[n-j-1]])
if st.query(l+1,r)>=j-i+1:
res += 2
i = j+1
return res+(i<=n-1-i)
|