首页 > 编程语言 >Manacher算法学习笔记

Manacher算法学习笔记

时间:2023-06-19 20:11:29浏览次数:57  
标签:Manacher texttt mid 笔记 算法 mx 回文

Manacher算法是什么

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

问题引入

P3805:【模板】manacher 算法

题目大意

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

算法

记录

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

算法推到

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

  • 当 \((i-1)-q_{i-1}+1>i-q_i+1\) 时,即以 \(i-1\) 为中心的回文串被 \([i-p_i+1,i+p_i-1]\) 所包含在内。
  • 当 \((i-1)-q_{i-1}+1\le i-q_i+1\) 时,即以 \(i-1\) 为中心的回文串并没有被 \([i-p_i+1,i+p_i-1]\) 所包含在内。

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

实现

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

\[\begin{cases}1(i>mx) \\ \min(mx-i+1,p_{mid*2-i})(i\le mx)\end{cases} \]

在判断 \(a_{i+p_i}\) 是否与 \(a_{i-p_i}\) 相同,相同就扩张 \(p_i\)。
然后在尝试用 \(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;
}

标签:Manacher,texttt,mid,笔记,算法,mx,回文
From: https://www.cnblogs.com/OIerBoy/p/17486508.html

相关文章

  • 【React工作记录一百一十四】前端小知识点扫盲笔记记录12
    前言我是歌谣放弃很容易但是坚持一定很酷微信公众号关注前端小歌谣带你进入前端巅峰交流群今天继续对前端知识的小结手写instanceOf~<!DOCTYPEhtml><htmllang="en"> <head> <metacharset="UTF-8"/> <metahttp-equiv="X-UA-Compatible"content="IE=edge&......
  • 【React工作记录一百一十五】前端小知识点扫盲笔记记录13
    前言我是歌谣放弃很容易但是坚持一定很酷微信公众号关注前端小歌谣带你进入前端巅峰交流群今天继续对前端知识的小结数组去重的方式<!DOCTYPEhtml><htmllang="en"> <head> <metacharset="UTF-8"/> <metahttp-equiv="X-UA-Compatible"content="IE=edge"......
  • 算法题总结-均等划分
    原题https://leetcode.cn/problems/partition-to-k-equal-sum-subsets/submissions/给定一个整数数组nums和一个正整数k,找出是否有可能把这个数组分成k个非空子集,其总和都相等。[1<=k<=len(nums)<=16]输入示例nums=[4,3,2,3,5,2,1],k=4输出示例True......
  • vim快捷键操作笔记
    vim快捷键操作笔记: vim打开文件快捷方式: vima.txt 打开或新建一个文件,并将光标置于第一行的首部 vim+a.txt 打开文件,并将光标置于最后一行的首部 vim+4a.txt 打开文件,并将光标位置于第4行首部 vim+/asdfa.txt 打开文件,并将......
  • 代码随想录算法训练营第十一天| 239. 滑动窗口最大值 347.前 K 个高频元素
    239.滑动窗口最大值 难点:1,想好怎么快速找到区块内的最大数值,往常使用的是在遍历一次,但是是O(m*n)思路:1,使用单调队列,所有的数值都必须是从大到小,2,用队列保持必要的顺序,而且对于大于K的循环,每次都要求poppush这两个操作代码:1voidpop(deque<int>&slidingWin......
  • VsCode配置PHP断点调试环境笔记
    PHPStudy_Pro8.1.1.2VsCode1.51.1PHP7.4.3NTSPHP_Xdebug-2.9.8-7.4-vc15-nts-x86_64首先查看当前环境的phpinfo信息根据phpinfo的信息选择对应的XDebug进行下载:https://xdebug.org/download推荐使用:https://xdebug.org/wizard,将phpinfo的信息全选复制到这里进行分析,然后下......
  • Android-kotlin学习笔记(一)配置/入门
    1.配置Kotlin开发插件,点击File菜单,选择Settings,选择Plugins,会显示扩展的插件;2.然后选择Browserepositories…,搜索栏目中搜索Kotlin即可,点击Install就行,大小50多M,速度很快的然后安装完成后,重启AndroidStudio3.在项目的build.gradle中配置Kotlin版本:ext.kotlin_version='1.2.......
  • 算法与数据结构Day03——平衡二叉树的根
    #include<stdio.h>#include<stdlib.h>typedefstructnode*AVLTree;structnode{intData;AVLTreeLeft;AVLTreeRight;};intHigh(AVLTreeT){if(!T)return0;intleft=High(T->Left)+1;intright=High(T->......
  • Java 编码(一)Java实现SHA256算法
    本文实例讲述了JavaSHA-256加密的两种实现方法。分享给大家供大家参考,具体如下:参考文献 Java实现SHA256算法-自学java的小陈-博客园(cnblogs.com)1、利用Apache的工具类实现加密:maven:<dependency><groupId>commons-codec</groupId><artifactId>commons-codec</......
  • Android中高级开发进阶必备资料(附:PDF+视频+源码笔记)
    前言Android开发学习过程中要掌握好基础知识,特别是java语言的应用,然后逐步提升开发者在学习过程中遇到的一些细致化的问题,把一些难点进行解决,在开发过程中把容易出现的一些难点进行合理化控制,避免在程序生成产品后出现问题,从而导致崩溃,这是非常重要的一点。架构师筑基必备技能作为......