info: { blogIcon: 'https://cdn.luogu.com.cn/upload/usericon/765493.png', blogUser: 'OIerBoy', blogAvatar: 'https://cdn.luogu.com.cn/upload/usericon/765493.png', blogStartDate: '2023-04-22', webpageTitleOnblur: '(o?v?)ノ Hi', webpageTitleOnblurTimeOut: 1000, webpageTitleFocus: '(*′?`*) 欢迎回来!', webpageTitleFocusTimeOut: 1000, webpageIcon: "https://cdn.jsdelivr.net/gh/BNDong/Cnblogs-Theme-SimpleMemory@master/img/webp/blog_logo.webp", { enable: true, // 是否开启日/夜间模式切换按钮 auto: { // 自动切换相关配置 enable: false, // 开启自动切换 dayHour: 5, // 日间模式开始时间,整数型,24小时制 nightHour: 19 // 夜间模式开始时间,整数型,24小时制 } } },

Manacher算法学习笔记

Manacher算法是什么

Manacher算法就是马拉车。
Manacher算法就是用于解决回文子串的个数的。

问题引入

P3805:【模板】manacher 算法

题目大意

给出一个只由小写英文字符 a,b,c,y,z 组成的字符串 S ,求 S 中最长回文串的长度。

算法

记录

为了找到最长的回文串的长度,我们首先就要考虑如何去把每一个回文串表示出来。
因为是回文的,所以我们可以用 pi 来表示。
其中 i 表示回文串的中心,pi 表示以第 i 个字符为中心的回文串的最长的回文串的半径。
但是这样我们只能表示奇数长度的回文串,而偶数回文串就不能解决。

算法推导

但是一个 S 的回文串个数最坏可能是 n2 级别的,会 TLE。
那么我们该如何快速得到每个以 i 为中心的最长的长度呢?
就像做 DP 题目一样,考虑类似 DP 的转移。
考虑如何通过 pi 来得到 pi+1
我们用一幅图来生动形象的体会一下:
image
这里我们就可以清晰的看到通过 pi 得到 pi+1 的两种。

  • (i1)qi1+1>iqi+1 时,即以 i1 为中心的回文串被 [ipi+1,i+pi1] 所包含在内。
  • (i1)qi1+1iqi+1 时,即以 i1 为中心的回文串并没有被 [ipi+1,i+pi1] 所包含在内。

第一种情况是很好办的,因为 i+1i1i 为中心对称,直接 pi+1=pi1
但是第二种情况就不好解决了,因为这就意味着我们似乎是要在继续判断 pi+1 的最大值,好像如果运气不好的话时间复杂度就会达到 O(n2)
这时就需要考虑单调性了,i 就可以不是 i+1 的前一个点,而可能是在 1i 中的一个点
想象一下,当出现第二种情况时,i+1 就必须要用 O(n) 来暴力得到 pi+1,但是如果 pi+1 覆盖了整个 [1,n] 的话,后面的 i+2n 就都会被 pi+1 所覆盖了。
即可以直接 O(1) 得到答案,时间复杂度也就是 O(n)
所以我们可以得到结论,Manacher 的时间复杂度具有单调性,是单调不下降的

实现

为了确保它的单调性,我们就需要一个 mid 来记录回文覆盖最大的点的下标,mxmid 回文串的左端点。
pi 的初始值就是:

{1(i>mx)min(mxi+1,pmid2i)(imx)

在判断 ai+pi 是否与 aipi 相同,相同就扩张 pi
然后在尝试用 i 去更新 mid,mx 就可以了。

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;
}
posted @   OIerBoy  阅读(59)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 单线程的Redis速度为什么快?
· 展开说说关于C#中ORM框架的用法!
· Pantheons:用 TypeScript 打造主流大模型对话的一站式集成库
· SQL Server 2025 AI相关能力初探
· 为什么 退出登录 或 修改密码 无法使 token 失效
点击右上角即可分享
微信分享提示