构建了回文树,求出了以每个位置结尾的回文串的数量
#include <bits/stdc++.h>
using namespace std;
#define ll long long
ll len[2000005],trie[2000005][26],num[2000005],fail[2000005];
char str[2000005];
ll size=0,now=0,ans=0,tot=1;
ll get_fail(ll x,ll y)//寻找最大符合要求的后缀字串的起始位置
{
while(y-len[x]-1<0||str[y-len[x]-1]!=str[y]) x=fail[x];
return x;
}
int main()
{
scanf("%s",str);
size=strlen(str);
fail[1]=0,fail[0]=1,tot=1,len[1]=-1,now=0;//初始化
for(int i=0;i<size;i++)
{
//if(i>=1) str[i]=(str[i]+ans-97)%26+97;//洛谷p5496
ll pos=get_fail(now,i);
if(!trie[pos][str[i]-'a'])//未找过这个回文串
{
fail[++tot]=trie[get_fail(fail[pos],i)][str[i]-'a'];//为新节点构建fail
trie[pos][str[i]-'a']=tot;
len[tot]=len[pos]+2;
num[tot]=num[fail[tot]]+1;
}
now=trie[pos][str[i]-'a'];
ans=num[now];
printf("%lld ",ans);//每个位置结尾的回文串数量
}
return 0;
}