回文自动机模板

构建了回文树,求出了以每个位置结尾的回文串的数量

#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;
} 

  • 7
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值