Manacher算法学习笔记
Manacher算法是什么
Manacher算法就是马拉车。
Manacher算法就是用于解决回文子串的个数的。
问题引入
题目大意
给出一个只由小写英文字符
算法
记录
为了找到最长的回文串的长度,我们首先就要考虑如何去把每一个回文串表示出来。
因为是回文的,所以我们可以用
其中
但是这样我们只能表示奇数长度的回文串,而偶数回文串就不能解决。
算法推导
但是一个
那么我们该如何快速得到每个以
就像做 DP 题目一样,考虑类似 DP 的转移。
考虑如何通过
我们用一幅图来生动形象的体会一下:
这里我们就可以清晰的看到通过
- 当
时,即以 为中心的回文串被 所包含在内。 - 当
时,即以 为中心的回文串并没有被 所包含在内。
第一种情况是很好办的,因为
但是第二种情况就不好解决了,因为这就意味着我们似乎是要在继续判断
这时就需要考虑单调性了,
想象一下,当出现第二种情况时,
即可以直接
所以我们可以得到结论,Manacher 的时间复杂度具有单调性,是单调不下降的。
实现
为了确保它的单调性,我们就需要一个
在判断
然后在尝试用
Code
#include<bits/stdc++.h>
#define N 12000005
#define int long long
using namespace std;
int n,mx=1,mid,ans,p[N<<2];
char a[N<<2],s[N<<1];
signed main(){
cin>>s+1;
n=strlen(s+1);
a[0]='$';
a[1]='#';
for(int i=1;i<=n;i++)
a[i<<1]=s[i],
a[(i<<1)+1]='#';
n=(n<<1)+2;
a[n]='@';
for(int i=1;i<=n;i++){
if(i<=mx)p[i]=min(mx-i+1,p[mid*2-i]);
else p[i]=1;
while(a[i+p[i]]==a[i-p[i]])++p[i];
if(i+p[i]>mx)mx=i+p[i]-1,mid=i;
ans=max(ans,p[i]);
}
cout<<ans-1;
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 单线程的Redis速度为什么快?
· 展开说说关于C#中ORM框架的用法!
· Pantheons:用 TypeScript 打造主流大模型对话的一站式集成库
· SQL Server 2025 AI相关能力初探
· 为什么 退出登录 或 修改密码 无法使 token 失效