从题目讲起
我们定义2个字符串的相似度等于两个串的相同前缀的长度。例如 “abc” 同 “abd” 的相似度为2,”aaa” 同 “aaab” 的相似度为3。
给出一个字符串S,计算S同他所有后缀的相似度之和。例如:S = “ababaa”,所有后缀为:
ababaa 6
babaa 0
abaa 3
baa 0
aa 1
a 1
S同所有后缀的相似度的和 = 6 + 0 + 3 + 0 + 1 + 1 = 11
想法
作为一个拿SAM模板随便刷水的人,这道题很明显反着插,求出每个状态right集合大小,从整个串的last一直沿着parent边往上跑,统计就行了。
结果爆了空间
没办法学扩展KMP
给定目标串S[1..m]和模板串T[1..n],求出目标串每个后缀与模板串的匹配长度。
设这个长度ext[i]。
假设现在1~i-1的ext都求出来了,现在要求i的,怎么做呢?
引入一个新数组next[i],表示T[i..n]与T的最大匹配长度。
设p=max(ext[a]+a-1),即之前最长匹配到S的哪里了。p取最大时,设位置为a。
整理一下:目前T[1..p-a+1]=S[a..p](①),那么T[i-a+1..p-a+1]=S[i..p](②).
我们要求S[i..m]和T[1..n]最长匹配,实质上,S[i..p]这一段我们是不用和T[1..p-