问题

什么是回文串,如果一个字符串正着度读和反着读是一样的,这个字符串就被称为回文串。

such as

noon level aaa bbb

既然有了回文,那就要有关于回文的问题,于是就有了——

最长回文子串:给定一个字符串,求它的最长回文子串长度。

暴力

找出所有的子串,遍历每个子串判断他们是否为回文串。

时间复杂度\(O(n^3)\)

优化

因为回文串是对称的,根据这个性质,枚举每个位置,找在这个位置上能扩展到的最长回文串。

时间复杂度\(O(n^2)\)

Manacher算法

打开洛谷的模板题发现数据范围是\(10^7\),还是不能过,怎么办。

先分析优化后的暴力的不足。

1.对于长度为奇数的回文和长度为偶数的回文,它们的对称轴是不一样的,要分类讨论。

2.有些子串会被访问多次。

比如 :

char: a b a b a b a ...
 i : 0 1 2 3 4 5 6 ...

我们枚举到第\(i\)位:

\(i==3\)时第一个"aba"被遍历了一次;

\(i==4\)时第一个"aba"又被遍历了一次;

\(i==5\)时第一个"aba"双被遍历了一次;

.......……

\(i==len/2\)时第一个"aba"又双叒叕被遍历了一次。

处理字符串长度的奇偶性带来的对称轴不确定问题

如果字符串的长度都是奇数就好办了。

处理原来的字符串,在收尾和所有空隙插入一个相同的无关的字符,插入后原字符串中是回文的子串还是回文串,不是回文的子串的依然不是。

但字符串的长度都变成了\(2*len+1\),都成了奇数.

为什么是\(2*len+1\),因为有\(len-1\)个空,两边又分别插入了\(2\)个,加起来等于\(len+(len-1)+2=2*len+1\)。

如:

长度为奇数的字符串

ababa ---> @#a#b#a#b#a#

长度为偶数的字符串

1221 ---> @#1#2#2#1#

'@'用来防止数组越界

找最长回文串

回文半径:把一个回文串中最左或最右位置的字符到其对称轴的距离称为回文半径

在Manacher算法中,我们用\(p[i]\)表示第\(i\)个字符的回文半径

    char : # a # b # c # b # a #
p[i] : 1 2 1 2 1 6 1 2 1 2 1
p[i] - 1 : 0 1 0 1 0 5 0 1 0 1 0
i : 1 2 3 4 5 6 7 8 9 10 11

显然,最大的\(p[i]-1\)就是答案

显然这个结论非常不显然,单从数值上看的话,插入完字符之后对于一个回文串的长度为原串长度*2+1,等于这个回文串回文半径*2+1,显然相等。

这样我们的问题就转换成了怎样快速的求出\(p\)数组

在这里我们利用回文串的对称性扩展回文串,p[i]不再直接赋值为1,而是根据之前求出的p[j],0<j<i。

我们用\(mx\)表示所有字符产生的最大回文子串的最大右边界,\(id\)表示产生这个最大右边界的对称轴的位置。

为什么要维护这些东西,因为我们要利用回文串的对称性来更新当前位置的值,维护了右边界(mx)后就可以直接判断当前位置是否可以直接利用对称性来更新(因为之前找到的回文串最右端就是到\(mx\),超出\(mx\)的话就不能利用对称性来更新了);\(id\)是对称轴,用来求关于\(i\)对称的位置\(j\)。



中间的#懒得画了就

如图,假设我们已经求出了\(p[1...7]\),当\(i<mx\)时,因为\(id\)被更新过了,而\(i\)是\(id\)之后的位置,第\(i\)个字符一定落在\(id\)的右边。这时我们关心的还是\(i\)是在\(mx\)的左边还是右边。

以下内容为了方便我们定义:

串\(i\)表示以\(i\)为对称轴的回文串(用红色的箭头表示);

串\(j\)表示以\(j\)为对称轴的回文串(用蓝色的箭头表示);

串\(id\)表示以\(id\)为对称轴的回文串(用绿色的箭头表示);

情况1:i < mx

如上图,利用回文串的性质,对于\(i\),我们可以找到一个关于\(id\)对称的位置\(j=id*2-i\),进行加速查找

但在这里又细分为了三种情况

(1)



显然此时\(p[i]=p[j]\)。

对于这种情况,串\(i\)不可以再向两边扩张。

如果可以向两边扩张的话,\(p[j]\)也可以再向两边扩张,而\(p[j]\)已经确定了,所以串\(i\)不向两边扩张。

(2)



显然此时\(p[i]=p[j]\)

与(1)不同的是,串\(i\)是可以再向两边扩张的。

应该很显然,一比划就知道。

(3)



此时\(p[i]=mx-i\)

这时我们只能确定串\(i\)在\(mx\)以内的部分是回文的,并不能确定串\(i\)和串\(j\)相同。

同样,这时我们的串\(i\)是不可以再向两端扩张的。



如果串\(i\)可以扩张,如图,则\(d=c\),根据对称性\(c=b\),又因为\(a=b\),所以\(a=d\),可以看到,串\(id\)可以继续扩张,因为\(p[id]\)已经固定了,所以串\(i\)不可以继续扩张

情况2:i >= mx



这时之前记录的信息都用不上了,于是\(p[i]=1\)。

综上所述

    if (i < mx) p[i] = min(p[id * 2 - i], mx - i);   //情况1
else p[i] = 1; //情况2
while (str[i + p[i]] == str[i - p[i]]) p[i]++; //暴力扩展
if (p[i] + i > mx) mx = p[i] + i, id = i; //更新id位置
}

代码

P3805 【模板】manacher算法

#include <bits/stdc++.h>
using namespace std;
const int N = 22000010;
char s[N];
char str[N];
int p[N];
int init() {
int len = strlen(s);
str[0] = '@', str[1] = '#';
int j = 2;
for (int i = 0; i < len; ++i) str[j++] = s[i], str[j++] = '#';
str[j] = '\0';
return j;
}
int manacher() {
int ans = -1, len = init(), mx = 0, id = 0;
for (int i = 1; i < len; ++i) {
if (i < mx) p[i] = min(p[id * 2 - i], mx - i);
else p[i] = 1;
while (str[i + p[i]] == str[i - p[i]]) p[i]++;
if (p[i] + i > mx) mx = p[i] + i, id = i;
ans = max(ans, p[i] - 1);
}
return ans;
} int main() {
cin >> s;
cout << manacher();
return 0;
}

Manacher算法详解的更多相关文章

  1. 算法进阶面试题01——KMP算法详解、输出含两次原子串的最短串、判断T1是否包含T2子树、Manacher算法详解、使字符串成为最短回文串

    1.KMP算法详解与应用 子序列:可以连续可以不连续. 子数组/串:要连续 暴力方法:逐个位置比对. KMP:让前面的,指导后面. 概念建设: d的最长前缀与最长后缀的匹配长度为3.(前缀不能到最后一 ...

  2. 经典算法 Manacher算法详解

    内容: 1.原始问题   =>O(N^2) 2.Manacher算法   =>O(N) 1.原始问题 Manacher算法是由题目“求字符串中长回文子串的长度”而来.比如 abcdcb 的 ...

  3. [转] Manacher算法详解

    转载自: http://blog.csdn.net/dyx404514/article/details/42061017 Manacher算法 算法总结第三弹 manacher算法,前面讲了两个字符串 ...

  4. hdu3068之manacher算法+详解

    最长回文 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submi ...

  5. manacher算法 详解+模板

    manacher算法可以解决字符串的回文子串长度问题. 个人感觉szy学长讲的非常好,讲过之后基本上就理解了. 那就讲一下个人的理解.(参考了szy学长的ppt) 如果一个回文子串的长度是偶数,对称轴 ...

  6. manacher算法详解+模板 P3805

    前言: 记住manacher是一个很简单的算法. 首先我们来了解一下回文字串的定义:若一个字符串中的某一子串满足回文的性质,则称其是回文子串.(注意子串必须是连续的,而子序列是可以不连续的) 那么若给 ...

  7. BM算法  Boyer-Moore高质量实现代码详解与算法详解

    Boyer-Moore高质量实现代码详解与算法详解 鉴于我见到对算法本身分析非常透彻的文章以及实现的非常精巧的文章,所以就转载了,本文的贡献在于将两者结合起来,方便大家了解代码实现! 算法详解转自:h ...

  8. kmp算法详解

    转自:http://blog.csdn.net/ddupd/article/details/19899263 KMP算法详解 KMP算法简介: KMP算法是一种高效的字符串匹配算法,关于字符串匹配最简 ...

  9. 机器学习经典算法详解及Python实现--基于SMO的SVM分类器

    原文:http://blog.csdn.net/suipingsp/article/details/41645779 支持向量机基本上是最好的有监督学习算法,因其英文名为support vector  ...

随机推荐

  1. 解决注册并发问题并提高QPS

    前言:前面在本地的windows通过apache的ab工具测试了600并发下“查询指定手机是否存在再提交数据”的注册功能会出现重复提交的情况,并且在注册完成时还需要对邀请人进行奖励,记录邀请记录,对该 ...

  2. SuperMap iClient 如何使用 WMTS 地图服务(转载)

    原文链接: WMTS服务初步理解与读取 https://blog.csdn.net/supermapsupport/article/details/76806670 SuperMap iClient ...

  3. iOS 设置View阴影

    iOS 设置View投影 需要设置 颜色 阴影半径 等元素 UIView *shadowView = [[UIView alloc] init]; shadowView.frame = CGRectM ...

  4. Linux AIDE(文件完整性检测)

    一.AIDE的概念 AIDE:Advanced Intrusion Detection Environment,是一款入侵检测工具,主要用途是检查文档的完整性.AIDE在本地构造了一个基准的数据库,一 ...

  5. 我的第一个python web开发框架(22)——一个安全小事故

    在周末的一个早上,小白还在做着美梦,就收到了小美的连环追魂call,电话一直响个不停. 小白打着哈欠拿起电话:早上好美女. 小美:出事了出事了,我们公司网站一早访问是一片空白,什么内容都没有了,你赶急 ...

  6. 戏说春秋_i春秋 writeup

    <戏说春秋>第一关 图穷匕见 题目: 解:用winhex打开,拉到最后可发现一段编码 放到解密网站上解码. <戏说春秋>第二关 纸上谈兵 解:文中没有明确指出问题,也没有给出线 ...

  7. sqlalchemy外键的一些东西

    sqlalchemy中让MySQL支持中文字符 engine = create_engine("mysql+pymysql://root:mysql8@localhost/mysqltest ...

  8. 【Python 17】B分R计算器1.0(数值类型)

    1.案例描述 基础代谢率(BMR):我们安静状态下(通常为静卧状态)消耗的最低热量,人的其他活动都建立在这个基础上. 计算公式: BMR(男) = (13.7*体重kg)+(5.0*身高cm)-(6. ...

  9. CSS伪元素:before/CSS伪元素:before/:after content 显示Font Awesome字体图标:after content 显示Font Awesome字体图标

    HTML <a href="javascript:volid(0);"><i class="icon-table"></i> ...

  10. rm: cannot remove ‘overlay/’: Device or resource busy

    umount /var/lib/docker/overlay #取消挂载就可以啦 rm -rf overlay/