求最长回文串,如果是暴力的方法的话,会枚举每个字符为中心,然后向两边检测求出最长的回文串,时间复杂度在最坏的情况下就是0(n^2),为什么时间复杂度会这么高,因为对于每一个作为中心的字符的检测是独立的,没有充分利用前面比较过信息,这就类似暴力求字符串的匹配最糟糕的情况下是0(n*m),然后通过预处理的信息把时间复杂度降低也就是kmp算法;

MANACHER算法:

先假设所有回文串都是以某个字符为中心的,即回文串的长度都是奇数;

lc[ i ]保存的是以位置i的字符为中心的最长回文串到最右边的距离

先假设以知lc[i],(0<=i<x)求lc[x]

设p=k+lc[k]-1,k是使p最大的i的取值,如下图

如果x>p那么直接以x为中心进行检测,并更新k;

如果x<=p那么对于以k为中心x的对称点就是j,并且lc[j]的值已经知道了,

If (lc[j]<p-x+1) lc[x]=lc[j]

因为s[j-1]!=s[j+1], s[j-1]=s[x-1],s[j+1]=s[x+1],所以s[x-1]!=s[x+1]如图:

If (lc[j]>=p-x+1) 那么lc[x]至少是lc[j] 如图:

但对于蓝色位置是不是还需要检测

时间复杂度分析:因为对于每一位s[i]都只被检测了一次,所以是o(n);

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

现在回到最原先的问题,因为回文串有可能是“1221”这种长度是偶数的,MANACHER算法提供了一个构造的方法可以统一这两种情况:在两个字符之间插入一个没有出现过的字符,如‘#’,那么”1221”->”#1#2#2#1#”这样所有的回文串长度都是奇数的了,为了处理方便再在该字符串前面加一个‘$’字符,这样字符串就是$#1#2#2#1#

求出lc[i]后,对于以s[i]为中心的字符串,如果s[i]==’#’,那么len=(lc[i]-1)/2*2,因为lc[i]-1肯定是偶数(因为该回文串的最左和最右肯定是’#’)所以len=lc[i]-1;

如果s[i]!=’#’,那么len=(lc[i] - 1)/2*2+1;因为lc[i]-1肯定是奇数,所以len=lc[i]-2+1=lc[i]-1;

所以最后的解就是最大lc[i]-1;

 void Manacher(char *s){
s1[]='$';
int nn=strlen(s),c=;
for (int i=;i<nn;i++){
s1[c++]='#';
s1[c++]=s[i];
}s1[c++]='#';s1[c++]='\0';
// cout<<s<<endl<<s1<<endl; lc[]=;lc[]=;
int k=, p, j;
for (int i=;i<c;i++){
p=k+lc[k]-;
if (p<i){
int t=;
while (i-t>= && i+t<c && s1[i-t]==s1[i+t]) t++;
lc[i]=t; k=i;
}else {
j=*k-i;
if (lc[j]<p-i+) lc[i]=lc[j];
else {
int t=p-i+;
while (i-t>= && i+t<c && s1[i-t]==s1[i+t]) t++;
lc[i]=t; k=i;
}
}
}
int ret=;
for (int i=;i<c;i++){
//cout<<lc[i]<<" ";
if (lc[i]->ret) ret=lc[i]-;
}//cout<<endl;
printf("%d\n",ret); }

MANACHER---求最长回文串的更多相关文章

  1. Manacher算法 - 求最长回文串的利器

    求最长回文串的利器 - Manacher算法 Manacher主要是用来求某个字符串的最长回文子串. 不要被manacher这个名字吓倒了,其实manacher算法很简单,也很容易理解,程序短,时间复 ...

  2. Manacher's Algorithm 马拉车算法(求最长回文串)

    作用:求一个字符串中的最长子串,同时还可以求所有子串的长度. 题目链接: https://vjudge.net/contest/254692#problem/B 最长回文串长度的代码: int Man ...

  3. hdu 3068 最长回文 (Manacher算法求最长回文串)

    参考博客:Manacher算法--O(n)回文子串算法 - xuanflyer - 博客频道 - CSDN.NET 从队友那里听来的一个算法,O(N)求得每个中心延伸的回文长度.这个算法好像比较偏门, ...

  4. hdu 3068 最长回文 【Manacher求最长回文子串,模板题】

    欢迎关注__Xiong的博客: http://blog.csdn.net/acmore_xiong?viewmode=list 最长回文                                 ...

  5. Manacher(最长回文串)

    http://acm.hdu.edu.cn/showproblem.php?pid=3068 最长回文 Problem Description 给出一个只由小写英文字符a,b,c...y,z组成的字符 ...

  6. Manacher算法,最长回文串

    给你10000长度字符串,然你求最长回文字串,输出长度,暴力算法肯定超时 #include <iostream> #include <string> #include < ...

  7. Manacher 计算最长回文串

    转自 http://blog.sina.com.cn/s/blog_3fe961ae0101iwc2.html 寻找字符串中的回文,有特定的算法来解决,也是本文的主题:Manacher算法,其时间复杂 ...

  8. Manacher 求最长回文子串算法

    Manacher算法,是由一个叫Manacher的人在1975年发明的,可以在$O(n)$的时间复杂度里求出一个字符串中的最长回文子串. 例如这两个回文串“level”.“noon”,Manacher ...

  9. POJ 3376 Finding Palindromes(manacher求前后缀回文串+trie)

    题目链接:http://poj.org/problem?id=3376 题目大意:给你n个字符串,这n个字符串可以两两组合形成n*n个字符串,求这些字符串中有几个是回文串. 解题思路:思路参考了这里: ...

  10. Manacher求最长回文

    #1032 : 最长回文子串 时间限制:1000ms 单点时限:1000ms 内存限制:64MB 描写叙述 小Hi和小Ho是一对好朋友.出生在信息化社会的他们对编程产生了莫大的兴趣,他们约定好互相帮助 ...

随机推荐

  1. Android真机调试的时候logcat中无法输出调试信息的解决办法

    真机调试不输出日志到logcat的原因是手机厂商默认关闭了调试打印的功能,通过以下方法开启此方法. 下面以华为P6手机为例进行操作: 1.在拨号界面输入:*#*#2846579#*#* 进入测试菜单界 ...

  2. poj 3258 River Hopscotch(二分+贪心)

    题目:http://poj.org/problem?id=3258 题意: 一条河长度为 L,河的起点(Start)和终点(End)分别有2块石头,S到E的距离就是L. 河中有n块石头,每块石头到S都 ...

  3. 获取本机外网ip和内网ip

    获取本机外网ip //获取本机的公网IP public static string GetIP() { string tempip = ""; try { WebRequest r ...

  4. UVa 10820 (打表、欧拉函数) Send a Table

    题意: 题目背景略去,将这道题很容易转化为,给出n求,n以内的有序数对(x, y)互素的对数. 分析: 问题还可以继续转化. 根据对称性,我们可以假设x<y,当x=y时,满足条件的只有(1, 1 ...

  5. bzoj3931: [CQOI2015]网络吞吐量

    将最短路图找出来,跑maxflow即可.有注意到数据范围.然后输出的时候%dWA了三次QAQ... #include<cstdio> #include<cstring> #in ...

  6. JSOI2008最大数(线段树)

    注意到数列只增不减,而题目中又明确说道m<=200000;这样的数据规模线段树完全可以承受得了.所以我们可以事先建好一棵200000个子节点的线段树,然后求极值就好了. type node=re ...

  7. 查看ORACLE执行计划的几种常用方法

    SQL的执行计划实际代表了目标SQL在Oracle数据库内部的具体执行步骤,作为调优,只有知道了优化器选择的执行计划是否为当前情形下最优的执行计划,才能够知道下一步往什么方向. 执行计划的定义:执行目 ...

  8. 【JSP】JSP向MySQL写入|读出中文数据——乱码问题

    注意第14行

  9. Android Studio进行NDK编程

  10. mysql 外连接总结

    内连接: 只连接匹配的行左外连接: 包含左边表的全部行(不管右边的表中是否存在与它们匹配的行),以及右边表中全部匹配的行右外连接: 包含右边表的全部行(不管左边的表中是否存在与它们匹配的行),以及左边 ...