Manacher算法学习笔记

这篇具有很好参考价值的文章主要介绍了Manacher算法学习笔记。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

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}\)
我们用一幅图来生动形象的体会一下:
Manacher算法学习笔记
这里我们就可以清晰的看到通过 \(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\) 就可以了。文章来源地址https://www.toymoban.com/news/detail-490668.html

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算法学习笔记的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 为什么要学习算法

    我们每个人可能都会有过的经历: 是不是从学校开始,你就觉得数据结构难学,然后一直没认真学? 工作中,一遇到数据结构这个坑,你又发自本能地迅速避让,因为你觉得自己不懂,所以也不想深究,反正看起来无关大局? 当你想换工作面试,或者研究某个开源项目源码

    2024年02月01日
    浏览(60)
  • 算法通过村第十八关-回溯|青铜笔记|什么叫回溯(中篇)

    提示:阳光好的时候,会感觉还可以活很久,甚至可以活出喜悦。 --余秀华 回溯是非常重要的算法思想之一,主要解决一些暴力枚举也搞不定的问题(这里埋个坑💣)例如组合、分割、子集、棋盘等等。从性能角度来看回溯算法的效率并不是很高,但是对于暴力也解决不了

    2024年02月06日
    浏览(46)
  • 机器学习笔记 - 什么是多模态深度学习?

            人类使用五种感官来体验和解释周围的世界。我们的五种感官从五种不同的来源和五种不同的方式捕获信息。模态是指某事发生、经历或捕捉的方式。         人工智能正在寻求模仿人类大脑,终究是跳不出这具躯壳的限制。         人脑由可以同时处理

    2024年02月09日
    浏览(41)
  • openGauss学习笔记-01 什么是openGauss

    openGauss学习笔记-01 什么是openGauss openGauss是一款全面友好开放,携手伙伴共同打造的企业级开源关系型数据库。openGauss提供面向多核架构的极致性能、全链路的业务、数据安全、基于AI的调优和高效运维的能力。openGauss深度融合华为在数据库领域多年的研发经验,结合企业级场

    2024年02月12日
    浏览(34)
  • (学习笔记-调度算法)进程调度算法

    进程调度算法也称 CPU 调度算法,毕竟进程是由 CPU 调度的。 当 CPU 空闲时,操作系统就选择内存中标的某个 [就绪状态] 的进程,将其分配给 CPU。 什么时候会发生CPU调度呢?通常有以下情况: 当进程从运行状态转换到等待状态 当进程从运行状态转换到就绪状态 当进程从等待

    2024年02月11日
    浏览(42)
  • 《算法工程师带你去》读书笔记 什么是稀疏向量(向量的稀疏表示)

    对数据进行预处理时,一般需要对类别型特征进行编码: 序号编码 独热编码 二进制编码 其中独热编码用的是最多的。但是当类别数十分巨大时,独热编码是一个非常稀疏的向量,只有一个值不为0,其他值均为0。可以使用向量的稀疏表示来大大的 节省空间 ,并且目前大多

    2024年02月03日
    浏览(38)
  • (学习笔记-进程管理)什么是悲观锁、乐观锁?

    最底层的两种就是 [互斥锁和自旋锁],有很多高级的锁都是基于它们实现的。可以认为它们是各种锁的地基,所以我们必须清楚它们之间的区别和应用。 加锁的目的就是保证共享资源在任意时间内,只有一个线程访问,这样就可以避免多线程导致共享数据错乱的问题。 当已

    2024年02月11日
    浏览(42)
  • 机器学习笔记 - 什么是图注意力网络?

            顾名思义,图注意力网络是图神经网络和注意力层的组合。要理解图注意力网络,我们首先需要了解什么是注意力层和图神经网络。首先,我们将看一下对图神经网络和注意力层的基本理解,然后我们将重点介绍两者的结合。让我们看一下图神经网络。       

    2023年04月09日
    浏览(38)
  • 机器学习笔记 - 什么是keras-core?

            简而言之,Keras Core 是 Keras API 的新多后端实现,支持 TensorFlow、JAX 和 PyTorch。         可以使用如下命令简单安装         Keras 是一个用 Python 编写的用于深度学习的用户友好工具。它旨在与 AI 领域的另一个主要参与者TensorFlow一起使用。Keras Core 是 Keras 的新

    2024年02月14日
    浏览(37)
  • 算法设计与分析学习笔记之二分查找算法

    二分查找只适用于有序的顺序表,非严格递增或是非严格递减都行。 二分查找运用到了分治的思想,将整体逐渐分为许多个小的部分,让整体的解变为诸多小部分解的合成,要求整体可以分解,小部分的解汇合之后可以得到整体部分的解。 至此,结束。 如果你觉得这篇文章

    2024年02月09日
    浏览(46)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包