【算法】Manacher(马拉车)算法
原jekyll 2019-09-07
Manacher’s Alogrithm,中文名叫马拉车算法,是一位叫Manacher的人在1975年提出的一种算法,解决的问题是求最长回文子串,算法的神奇之处就在于将时间复杂度精进到了O(N)。还记得在两年前的四省赛中,有一道关于回文的题,题解就是用马拉车算法做解的,然而我们没有做出来。
01 由来
在求解最长回文子串时,一般的思路是以当前字符为中心向两边扩展寻找回文,但复杂度是O(N^2),那能不能将复杂度降低到线性?马拉车算法就是为此诞生的。
02 预处理
为了在处理字符串的时候不需要为字符串长度是奇数还是偶数而分别考虑,将对原始字符串进行处理,在每一个字符的左右两边都加上特殊字符(肯定不存在原始字符串中的字符),让字符串变成一个长度为奇数的字符串。如: abba --> #a#b#b#a#
03 计算最长回文子串长度
以字符串"arddrb"为例,将预处理后的新字符串"#a#r#d#d#r#b#",作为一个新的字符串arr,定义一个辅助数组Len,Len的长度与arr等长,用Len[i]来表示以arr[i]字符为中心的最长回文半径。
在等待Len数组计算出来之后,取Len数组中值最大的数所对应的下标,就是arr字符串中的所对应的字符为中心的回文串的半径,Len数组有一个特点:Len[i]-1的值,就是原字符串中该以该字符为中心的回文串的长度。以下为i、arr、Len对应的值
i 0 1 2 3 4 5 6 7 8 9 10 11 12
arr # a # r # d # d # r # b #
Len 1 2 1 2 1 2 5 2 1 2 1 2 1
04 计算回文子串起始索引
取出Len数组中最大的值的索引i后,应该如何由得到原字符串该字符的索引呢?继续以str=“arddrb"为例,有arr=”#a#r#d#d#r#b#",Len[6]=5,发现6-Len(6)=1,即i-Len[i]就是arr[i]字符在原始字符串中的下标。但以str=“aba"为例,arr=”#a#b#a#",Len[3]=4,3-Len[3]=-1,所以str[-1]将会溢出。为了避免奇回文溢出,所以在arr的首尾再分别添加一个特殊字符,如下
i 0 1 2 3 4 5 6 7 8 9 10
arr @ # c # a # b # a # $
Len 1 1 2 1 2 1 4 1 2 1 1
可以看出,对于b字符来说,6-Len(6)=2,可以得到最长回文子串的起始索引为(i-Len(i))/2。
05 计算Len数组
第三步和第四部都是以Len数组计算完成为前提来进行的,计算Len数组就是马拉车算法的主要工作了,还是以"arddrb"为例,
i 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14
arr @ # a # r # d # d # r # b # $
Len 1 1 2 1 2 1 2 5 2 1 2 1 2 1 1
定义两个变量Mi和R,Mi是所有回文子串中,能延伸到最右端位置的那个回文子串的中心点位置,R是该回文串能延伸到最右端的位置。
当i=7时,Len[i]=5,在以位置7位中心点的回文子串中,该回文串的右边界是位置12。
当i=12时,Len[i]=2,在以位置12位中心点的回文子串中,该回文串的右边界是位置14。
所以可以得到,R=Len[i]+i
具体编程时,从左往右计算数组Len,需要分以下情况
1) 当i <= R时,首先毫无疑问Len数组中i点之前的对应的值已经求出来了,利用回文的特点,只要找到i关于Mi点对称的点j,j=Mi* 2-i,i、j在以Mi为中心的回文串的范围内[L, R]:
- 如果Len[j] < R-i(同样是L到j的距离),说明以j为中心的回文串没有超出范围[L, R],所以,Len[i] = Len(j),如下图
- 如果Len[j] >= R-i,即j为中心的回文串的最左端超过L,如下图所示,所以Len[i] = Len[j]是不成立的,有回文串的特性可知,Len[i] 至少等于R-i,至于是否大于R-i,那还得需要从R+1开始一一匹配,直到失配为止,从而更新R和对应的中心点Mi以及Len[i]。
2) 当i > R时,如下图,这种情况没法用到回文串的特性,只能老实地去一一匹配。
代码如下(以leetcode第5题为例)
public String longestPalindrome(String s) {int mi = 0;int right = 0;int maxlength = 0;int maxpoint = 0;String temp = "@#";for(int i = 0; i < s.length(); i++){temp += s.charAt(i);temp += "#";}temp += "*";int[] p = new int[temp.length()];for(int i = 0; i < temp.length(); i++){p[i] = 0;}for(int i = 1; i < temp.length()-1; i++){p[i] = right > i? Math.min(p[2*mi-i], right - i) : 1;while(temp.charAt(i+p[i]) == temp.charAt(i-p[i])){p[i]++;}if(i + p[i] > right){right = i + p[i];mi = i;}if(maxlength < p[i]){maxlength = p[i];maxpoint = i;}}//(maxpoint - maxlength)/2为最长回文数的起始点,maxlength为最长回文数的长度return s.substring((maxpoint - maxlength)/2, (maxpoint - maxlength)/2 + maxlength - 1);
}
理解p[i] = right > i? Math.min(p[2* mi-i], right - i) : 1;
就将manacher理解的差不多了。
【算法】Manacher(马拉车)算法相关推荐
- 【HDU - 3068】最长回文(Manacher算法,马拉车算法求最长回文子串)
题干: 给出一个只由小写英文字符a,b,c...y,z组成的字符串S,求S中最长回文串的长度. 回文就是正反读都是一样的字符串,如aba, abba等 Input 输入有多组case,不超过120组 ...
- Manacher马拉车算法求最长回文子串
终于把马拉车算法搞明白了!赶紧记录一下. 这个算法用于查找一个字符串的最长回文子串 马拉车算法依次给数组p[i]赋值,马拉车算法的本质就是在每次给数组p[i] 赋值时尝试进行偷懒 例如,当要给p[6] ...
- Manacher(马拉车)算法—简略讲解
这是一篇菜鸡的算法小笔记!希望你喜欢! 前言 马拉车算法是用来查找一个字符串的最长回文子串的线性方法,是一个叫 Manacher 的人在 1975 年发明的,这个方法的最大贡献是在于将时间复杂度提升到 ...
- manacher马拉车算法
算法讲解 算法讲解1 #include <iostream> #define maxn 10e+6; using namespace std; char s[maxn],str[maxn* ...
- (manacher)马拉车算法专题题目
manacher算法用来求解回文串问题,时间复杂度为O(n). 不懂的先可以去练习下模板板子题,求最长回文串 传送门P3501 [POI2010]ANT-Antisymmetry 这一题他给的是一个新 ...
- Manacher (马拉车)算法
Manacher于1975年发现了一种线性时间算法,可以在列出给定字符串中从任意位置开 始的所有回文子串.同样的算法也可以在任意位置查找全部极大回文子串,并且时间复杂 度是线性的.那他是怎样实现的呢, ...
- 什么是Manacher(马拉车)算法-java代码实现
截止到目前我已经写了 500多道算法题,其中部分已经整理成了pdf文档,目前总共有1000多页(并且还会不断的增加),大家可以免费下载 下载链接:https://pan.baidu.com/s/1hj ...
- Manacher(马拉车算法)
对应letecode链接: https://leetcode-cn.com/problems/longest-palindromic-substring/ 题目描述: 给你一个字符串 s,找到 s 中 ...
- 最长回文 HDU - 3068(求最长回文串的长度【马拉车算法Manacher】)
马拉车算法 Manacher's Algorithm 是用来查找一个字符串的最长回文子串的线性方法,由一个叫 Manacher 的人在 1975 年发明的,这个方法的最大贡献是在于将时间复杂度提升到了 ...
最新文章
- 难以置信!LSTM和GRU的解析从未如此清晰(动图+视频)
- 为何学习新知识这么难?因为大脑可能比你想象中更死板
- 数据结构 -- 二叉树
- 【Groovy】MOP 元对象协议与元编程 ( 使用 Groovy 元编程进行函数拦截 | 重写 MetaClass#invokeMethod 方法拦截 JDK 中已经定义的函数 )
- VS2008文件操作出现问题
- Source Generator:C# 9 将迎来编译时元编程
- 使用Spring Roo进行快速云开发–第2部分:VMware Cloud Foundry
- 线程池原理及C语言实现线程池
- [图]美专家称人类可能永远无法飞出太阳系
- java基础—面向对象——变量的区别
- 【Qt串口调试助手】1.3 - 重写ComboBox下拉框的鼠标点击事件,实现点击下拉框扫描可用串口
- java整型_java整型数值
- axure rp编辑html模板,AxureRP教程—用模板封装UI标准
- LED透明屏为什么能透明?实现原理
- 期货大作手风云录 --简记
- mysql80004005错误_常见的80004005错误及其解决方法
- 你的大三,推荐做的几件事
- DFRobot高精度低功耗数字温湿度传感器SHT40
- 2021年跨境电商行业相关数据
- IDEA的下载及安装