Manacher 学习笔记
\(\\\)
\(Manacher\)
一种常用的字符串算法,用于处理一些回文字符相关的问题。
- 回文串:从前向后和从后向前输出一致。
- 回文中心:以这里开始,每次向外左右各扩展一个字符得到的回文串的中心。
- 回文半径:定义在字符串的一个字符或两个字符的间隙上,代表以这里为回文中心的最长回文串的半径,具体的说,如果位置\(i\)延申出的最长回文串区间为\([l,r]\),那么他的回文半径就是\(r-i+1\)。
\(Manacher\)最基本的操作,就是求出以一个字符串的每一个字符和间隙为回文中心,对应的回文半径。
\(\\\)
\(\\\)
具体操作
\(\\\)
首先考虑间隙这个问题。由于便于描述间隙的回文半径,\(Manacher\)使用了一种巧妙的转化:
在每一个间隙(包括第一个字符前和最后一个字符后)各插上一个特殊字符,即原字符串中没有出现过的字符。为了不影响原串的正反匹配,所加的字符都是一样的。为了避免头尾匹配越界,头尾各放置一个与上述提到过的所有字符均不同的字符,且这两个字符也要不同。
我的方式一般是正常插入#,两侧分别插入不同的中括号。形象化的表示:

注意两侧多添加的字符只是为了匹配不越界,并不用于统计回文半径。
\(\\\)
\(\\\)
关于还原回原串的回文长度问题我们后面再讲,先考虑最优秀的复杂度求出新串所有位置的回文半径是多少。
线性。设\(len_i\)表示新字符串中以\(i\)为回文中心的回文半径,做的时候维护两个变量\(Maxr\)和\(Maxp\),第一个代表当前已经处理过的回文中心,其创造的回文子串最远延申到的字符位置,第二个就是对应的回文中心。
然后考虑如何快速的得到当前要求的位置\(i\)的答案,若\(i\not=1\),显然有\(i>Maxp\)。
我们可以用对称性很快的找到\(i\)关于\(Maxp\)的对称点\(i'\),并且之前已经确定了\(len_{i'}\)。观察下面的两种情况。

线段都表示以对应点为回文中心的最长回文子串。这一情况中,对称点的子串范围再对称回来并没有超过\(Maxp\)的子串范围,因为红色的大串是回文的,所以两侧的情况应该相同,即\(i\)所对应的回文子串长度应该与\(i'\)相同,\(len_i=len_{i'}\)。

这一情况中,我们一定能保证,以红色左端点为起点,以\(i'\)为对称中心的对称点为终点的部分,再关于\(Maxp\)对称过去是\(i\)可延伸的一个回文子串,但是再长因为超过了当前\(Maxr\)的范围,所以不能确定。这一情况下,先令\(len_i=len_{i'}\),再一个一个位置的尝试向外扩展。
当前位置直接超过\(Maxr\)时,我们也视作第二种情况。
当前位置的右端点超过\(Maxr\)时,更新\(Maxr\)和\(Maxp\)。
for(R int i=1,p=0,mr=0;i<=n;++i){
len[i]=i>mr?1:min(mr-i+1,len[(p<<1)-i]);
while(s[i-len[i]]==s[i+len[i]]) ++len[i];
if(i+len[i]-1>mr){p=i;mr=i+len[i]-1;}
}
预处理部分就不放代码了,核心其实就这四行。
\(\\\)
\(\\\)
复杂度证明和一些推论
\(\\\)
粗略的复杂度证明。扫描是\(\text O(N)\)的,每一个第一种情况得到答案是\(\text O(1)\)的,第二种情况至多只会出现\(N\)次,且每一次需要扩展右端点的扫描距离之和等于\(N\),这种类似于双指针扫描的东西复杂度是线性的。
\(\\\)
\(\\\)
有一个特殊的性质,即一个字符在原字符串的意义下\((\)#号即代表一个间隙\()\)做回文中心,他延申出的最长回文子串的长度,等于转化之后的字符串中,他的回文半径\(-1\)。我们分两种情况讨论证明它。
- 转化后的字符为特殊字符\((\)#号\()\)。这一情况中得到的回文串,从这个#号开始的部分一定是形如#a#b....#z#的,注意到在不算中间字符时,右一半的#号和原串字符个数是一样的,左一半也是一样。因为中间字符不是原串字符,所以\(len-1\)就是以这个#号为回文中心,延伸出的最长回文子串的长度。
- 原串的字符。这一情况得到的回文串,从这个字符开始的部分一定是形如a#b#.....z#的形式,这一情况下不算中间字符时,右一半#号个数比原串字符个数多一,左一半也是这样,所以右一侧回文半径中所有的#号都拿左侧的换,还多一个#号,正好用回文中心换掉。
关于上面的“一定是形如”部分,粗略的说明可以理解成,每个字符一定是被两个#号围着的,没有任何一个原串字符两侧出现除掉#号以外的其他字符。
\(\\\)
\(\\\)
还有一个定理可以得出,是一个字符串最多只有\(N\)个本质不同的回文子串。
考虑终点相同的回文子串 , 可以发现短的回文子串在大的回文子串中,因此对称过去在前面出现过。所以以某一个位置为终点且第一次出现的回文子串最多只有一个。这也是\(Manacher\)时间复杂度有保证的原因,因为如果\(Maxr\)不更新,就不会出现本质不同的回文子串,前面已经出现过了。而每扩展一次\(Maxr\),最多新出现一个本质不同的回文子串。
\(\\\)
\(\\\)
一道例题
\(\\\)
给出一个长度为\(N\)字符串,每个回文的部分都是一个碎片(碎片之间可部分重合),求最少多少次拼合碎片能够得出原字符串,拼合的定义是只要不完全相同就可以连接在一起,如果头尾有相同的部分可以重合。如\(aba\)和\(aca\)连接起来,可以生成串\(abaaca\)或 \(abaca\)。
- \(N\in [5\times10^4]\),多组数据。
\(\\\)
\(Manacher\)板子。求出所有回文串在原字符串中的覆盖区间,就是最少线段完全覆盖问题,贪心即可。
关于为什么一定是最长回文子串的问题,考虑两个回文串想要重叠的部分,划给哪一侧另一侧都会失去那么长的长度,所以最长的回文子串可以代表所有子串的最优答案。
有一个化简是,注意到能覆盖一个字符,就一定能覆盖两侧的#号,所以直接统计在变化后的串每一个回文串的覆盖区间即可,不用还原回去。需要注意拼合的次数是总段数\(-1\)。
\(\\\)
#include<cmath>
#include<cstdio>
#include<cctype>
#include<string>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 50010
#define R register
using namespace std;
char s[N<<1],sr[N];
int p,mr,now,ans,slen,len[N<<1];
struct line{int l,r;}seg[N<<1];
inline bool cmp(line x,line y){return x.l<y.l;}
inline void init(){
p=mr=ans=0;
s[now=1]='#';
slen=strlen(sr);
for(R int i=0;i<slen;++i){s[++now]=sr[i];s[++now]='#';}
s[0]='['; s[now+1]=']';
}
inline void manacher(){
for(R int i=1;i<=now;++i){
len[i]=(i>mr)?1:min(mr-i+1,len[(p<<1)-i]);
while(s[i-len[i]]==s[i+len[i]]) ++len[i];
if(i+len[i]-1>mr){mr=i+len[i]-1;p=i;}
seg[i].l=i-len[i]+1; seg[i].r=i+len[i]-1;
}
}
inline void calc(){
sort(seg+1,seg+1+now,cmp);
for(R int i=1,nowr=1,tmp=0;i<=now;){
while(seg[i].l<=nowr&&i<=now) tmp=max(tmp,seg[i].r),++i;
if(tmp+1<=nowr) break;
++ans; nowr=tmp+1;
}
printf("%d\n",ans-1);
}
int main(){
while(scanf("%s",sr)!=EOF){init();manacher();calc();}
return 0;
}
Manacher 学习笔记的更多相关文章
- Manacher学习笔记
目录 code(伪) Manacher算法 可在 \(O(n)\)的时间内求出一个字符串以每个位置为中心的最长回文子串. 原理:根据之前预处理出的回文串长度求得新的回文串长度 我们可以通过在字符中加上 ...
- Manacher算法学习笔记 | LeetCode#5
Manacher算法学习笔记 DECLARATION 引用来源:https://www.cnblogs.com/grandyang/p/4475985.html CONTENT 用途:寻找一个字符串的 ...
- 学习笔记 - Manacher算法
Manacher算法 - 学习笔记 是从最近Codeforces的一场比赛了解到这个算法的~ 非常新奇,毕竟是第一次听说 \(O(n)\) 的回文串算法 我在 vjudge 上开了一个[练习],有兴趣 ...
- 【学习笔记】字符串—马拉车(Manacher)
[学习笔记]字符串-马拉车(Manacher) 一:[前言] 马拉车用于求解连续回文子串问题,效率极高. 其核心思想与 \(kmp\) 类似:继承. --引自 \(yyx\) 学姐 二:[算法原理] ...
- OI知识点|NOIP考点|省选考点|教程与学习笔记合集
点亮技能树行动-- 本篇blog按照分类将网上写的OI知识点归纳了一下,然后会附上蒟蒻我的学习笔记或者是我认为写的不错的专题博客qwqwqwq(好吧,其实已经咕咕咕了...) 基础算法 贪心 枚举 分 ...
- js学习笔记:webpack基础入门(一)
之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...
- PHP-自定义模板-学习笔记
1. 开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2. 整体架构图 ...
- PHP-会员登录与注册例子解析-学习笔记
1.开始 最近开始学习李炎恢老师的<PHP第二季度视频>中的“章节5:使用OOP注册会员”,做一个学习笔记,通过绘制基本页面流程和UML类图,来对加深理解. 2.基本页面流程 3.通过UM ...
- 2014年暑假c#学习笔记目录
2014年暑假c#学习笔记 一.C#编程基础 1. c#编程基础之枚举 2. c#编程基础之函数可变参数 3. c#编程基础之字符串基础 4. c#编程基础之字符串函数 5.c#编程基础之ref.ou ...
随机推荐
- T1075 明明的随机数 codevs
http://codevs.cn/problem/1075/ 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 白银 Silver 题目描述 Description 明明想在学校中 ...
- JFrame实现批量获取Android安装包安全证书MD5
今天遇到一个需求.获取全部apk的签名的MD5.以下是我使用Java SE实现的一个工具.贴出核心源码.希望给有须要的朋友有所帮助. 界面例如以下: 仅仅须要制定.apk文件所在的文件夹就可以,核心代 ...
- POJ 1386 Play on Words(有向欧拉通路 连通图)
题意 见下方中文翻译 每一个单词能够看成首尾两个字母相连的一条边 然后就是输入m条边 推断是否能构成有向欧拉通路了 有向图存在欧拉通路的充要条件: 1. 有向图的基图连通: 2. 全部点的出度和 ...
- Andriod开发技巧——Fragment的懒载入
我们在做应用开发的时候.一个Activity里面可能会以viewpager(或其它容器)与多个Fragment来组合使用.而假设每一个fragment都须要去载入数据.或从本地载入.或从网络载入,那么 ...
- vue—你必须知道的 js数据类型 前端学习 CSS 居中 事件委托和this 让js调试更简单—console AMD && CMD 模式识别课程笔记(一) web攻击 web安全之XSS JSONP && CORS css 定位 react小结
vue—你必须知道的 目录 更多总结 猛戳这里 属性与方法 语法 计算属性 特殊属性 vue 样式绑定 vue事件处理器 表单控件绑定 父子组件通信 过渡效果 vue经验总结 javascript ...
- VC 获取任务栏窗体的句柄
本文将介绍一个未公开的Win32 API函数:GetTaskmanWindow.利用它对Windows的任务栏进行操作. 这个函数返回拥有任务栏button的窗体句柄. 在微软的MSDN文档中. ...
- python 循环高级用法 [expression for x in X [if condition] for y in Y [if condition] ... for n in N [if condition] ]按照从左至右的顺序,分别是外层循环到内层循环
高级语法 除了像上面介绍的 [x ** 2 for x in L] 这种基本语法之外,列表推导式还有一些高级的扩展. 4.1. 带有if语句 我们可以在 for 语句后面跟上一个 if 判断语句,用于 ...
- B4197 [Noi2015]寿司晚宴 状压dp
这个题一开始想到了唯一分解定理,然后状压.但是显然数组开不下,后来想到每个数(n<500)大于19的素因子只可能有一个,所以直接单独存就行了. 然后正常状压dp就很好搞了. 题干: Descri ...
- bzoj2216
决策单调性+整体二分 这里就是j<k且kj劣于j,j不会再选,所以我们整体二分 pos是因为从L->R中这个是最优点,所以对于mid+1->r选pos之前肯定不优,l->mid ...
- 创建APP检查更新页
本文来源及参考:Create a check for updates page for your app. 这篇文章解释了如何创建一个简单的检查更新页,检查该用户已安装的应用程序的最新版本. 简介 这 ...