Problem Description
给出一个只由小写英文字符a,b,c...y,z组成的字符串S,求S中最长回文串的长度.
回文就是正反读都是一样的字符串,如aba, abba等
 
Input
输入有多组case,不超过120组,每组输入为一行小写英文字符a,b,c...y,z组成的字符串S
两组case之间由空行隔开(该空行不用处理)
字符串长度len <= 110000
 
Output
每一行一个整数x,对应一组case,表示该组case的字符串中所包含的最长回文长度.
 
Sample Input

aaaa abab
 
Sample Output

4 3

题意:模板题,求最长回文子串的长度,字符串长度有11W,可以用manacher算法,注意输入的时候如果写成while(scanf("%s", a))会tle,应该写成while(scanf("%s", a) == 1)

manacher(马拉车)算法:复杂度O(n)

作用:给出一个字符串,求它的最长回文子串(的长度),求具体的串可以在模板上改一改

1.将字符串之间插入串中不可能有的字符,如将a a b a b a a b处理成 $ # a  # a # b # a # b # a # a # b #,

一个回文串是对称的,既然对称,就一定有那个“对称轴”,对于长度为奇数的回文,它的对称轴是一个字符,对于长度为偶数的回文串,它的对称轴是两个字符。为了方便处理,将所有回文串的对称轴变为一个字符,可以如上插入不相关字符,这样连上刚插入的字符,原串就变成了奇数长度。 最前面的  '$' 是为了防止溢出(我也不太懂)

2.设数组Mp[i]为以i为中心的回文串的半径长度,

i          0  1  2  3  4  5  6  7  8  9 10 11 12 13 14 15 16 17

ma[i]   $  #  a  #  a  #  b  #  a  #  b  #   a   #   a   #   b   #

mp[i]   0  0  1  2  1  0  3  1  7  ...

如此可以观察到mp[i]也代表 原串中 以i为中心的回文串的长度 ! 所以最长回文串的长度就是mp数组中的最大值了

3.求mp数组

用i将整个ma数组遍历一遍,在遍历的过程中维护一个已知的最长回文串,这个回文串要求它的右边界最靠右、右边界相同的话中心最靠左,

以下转载于https://www.felix021.com/blog/read.php?2040

该算法增加两个辅助变量(其实一个就够了,两个更清晰)id和mx,其中 id 为已知的 {右边界最大} 的回文子串的中心,mx则为id+mp[id],也就是这个子串的右边界。

然后可以得到一个非常神奇的结论,这个算法的关键点就在这里了:如果mx > i,那么P[i] >= MIN(P[2 * id - i], mx - i)。就是这个串卡了我非常久。实际上如果把它写得复杂一点,理解起来会简单很多:

//记j = 2 * id - i,也就是说 j 是 i 关于 id 的对称点(j = id - (i - id))
if (mx - i > mp[j]) 
    mp[i] = mp[j];
else /* mp[j] >= mx - i */
    mp[i] = mx - i; // mp[i] >= mx - i,取最小值,之后再匹配更新。

当然光看代码还是不够清晰,还是借助图来理解比较容易。

当 mx - i > P[j] 的时候,以S[j]为中心的回文子串包含在以S[id]为中心的回文子串中,由于 i 和 j 对称,以S[i]为中心的回文子串必然包含在以S[id]为中心的回文子串中,所以必有 P[i] = P[j],见下图。

当 P[j] >= mx - i 的时候,以S[j]为中心的回文子串不一定完全包含于以S[id]为中心的回文子串中,但是基于对称性可知,下图中两个绿框所包围的部分是相同的,也就是说以S[i]为中心的回文子串,其向右至少会扩张到mx的位置,也就是说 P[i] >= mx - i。至于mx之后的部分是否对称,就只能老老实实去匹配了。

对于 mx <= i 的情况,无法对 P[i]做更多的假设,只能P[i] = 1,然后再去匹配了。

于是代码如下:

//输入,并处理得到字符串s
int mp[1000], mx = 0, id = 0;
memset(mp, 0, sizeof(mp));
for (i = 1; s[i] != '\0'; i++) {
    p[i] = mx > i ? min(mp[2*id-i], mx-i) : 1;
    while (s[i + mp[i]] == s[i - mp[i]]) mp[i]++;
    if (i + mp[i] > mx) {
        mx = i + mp[i];
        id = i;
    }
}
//找出mp[i]中最大的

整个manacher算法就是这些,另外贴一下该题目的代码,也是manacher的模板,模板取自上海大学kuangbin

#include<cstdio>
#include<cstring>
#include<algorithm> #define clr(a, x) memset(a, x, sizeof(a))
typedef long long ll;
typedef long double ld; using namespace std; const int maxn = 110011;
char ma[maxn * 2];
int mp[maxn * 2]; void manacher(char s[], int len){
int l = 0;
ma[l++] = '$';
ma[l++] = '#';
for(int i = 0 ; i < len; i++){
ma[l++] = s[i];
ma[l++] = '#';
}
ma[l] = 0;
int mx = 0, id = 0;
for(int i = 0; i < l; i++){
mp[i] = mx > i ? min(mp[2*id - i],mx - i): 1;
while(ma[i + mp[i]] == ma[i - mp[i]])mp[i]++;
if(i + mp[i] > mx){
mx = i + mp[i];
id = i;
}
}
} int main(){
char s[maxn];
while(scanf("%s", s) == 1){
int len = strlen(s);
manacher(s, len);
int ans = 0;
for(int i = 0; i < 2 * len + 2; i++)
ans = max(ans, mp[i] -1);
printf("%d\n", ans);
}
return 0;
}

HDU 3068 最长回文 (Manacher最长回文串)的更多相关文章

  1. hdu 3068 最长回文 manacher算法(视频)

    感悟: 首先我要Orz一下qsc,我在网上很难找到关于acm的教学视频,但偶然发现了这个,感觉做的很好,链接:戳戳戳 感觉这种花费自己时间去教别人的人真的很伟大. manacher算法把所有的回文都变 ...

  2. HDU - 3068 最长回文(manacher)

    HDU - 3068 最长回文 Time Limit: 2000MS   Memory Limit: 32768KB   64bit IO Format: %I64d & %I64u Subm ...

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

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

  4. HDU 3068 最长回文( Manacher模板题 )

    链接:传送门 思路:Manacher模板题,寻找串中的最长回文子串 /***************************************************************** ...

  5. HDU 3068 (Manacher) 最长回文

    求一个字符串的最长子串,Manacher算法是一种O(n)的算法,很给力! s2[0] = '$',是避免在循环中对数组越界的检查. 老大的代码: http://www.cnblogs.com/Big ...

  6. 最长回文子串(百度笔试题和hdu 3068)

    版权所有.所有权利保留. 欢迎转载,转载时请注明出处: http://blog.csdn.net/xiaofei_it/article/details/17123559 求一个字符串的最长回文子串.注 ...

  7. hdu 3068 最长回文(manachar求最长回文子串)

    题目连接:hdu 3068 最长回文 解题思路:通过manachar算法求最长回文子串,如果用遍历的话绝对超时. #include <stdio.h> #include <strin ...

  8. hdu_3068 最长回文(Manacher算法)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3068 最长回文 Time Limit: 4000/2000 MS (Java/Others)    M ...

  9. Manacher 算法(hdu 3068 && hdu 3294)

    今天打算补前晚 BC 的第二题,发现要用到能在 O(n) 时间求最大回文子串长度的 Manacher 算法,第一次听,于是便去百度了下,看了大半天,总算能看懂了其思想,至于他给出的代码模板我没能完全看 ...

  10. 【BZOJ-2342】双倍回文 Manacher + 并查集

    2342: [Shoi2011]双倍回文 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1799  Solved: 671[Submit][Statu ...

随机推荐

  1. ACM北大暑期课培训第二天

    今天继续讲的动态规划 ... 补充几个要点: 1. 善于利用滚动数组(可减少内存,用法与计算方向有关) 2.升维 3.可利用一些数据结构等方法使代码更优  (比如优先队列) 4.一般看到数值小的 (十 ...

  2. Map and HashMap

    1.1.1. Map 接口 java提供了一组可以以键值对(key-value)的形式存储数据的数据结构,这种数据结构称为Map.我们可以把Map看成一个多行两列的表格,其中第一列存放key,第二列存 ...

  3. C++Primer第五版 3.2.3节练习

    练习 3.6:编写一段程序,使用范围for语句将字符串内的所有字符用X代替. #include<iostream> #include<string> using namespa ...

  4. 【转】ArcGIS ADF 实时轨迹问题初步解决方案

    Web ADF 实时轨迹是指在Web客户端指定一资源项,并对资源项进行实进跟踪并绘制出轨迹图.实时绘制可采用Ajax实现服务端与客户端无刷新动态绘制,在.net2.0 框架下可轻易实现:通过客户端时钟 ...

  5. SQL Server2012高可用之日志传送测试

    (一)日志传送架构 (1.1)相关服务器 主服务器   :用于生产的服务器,上面运行这生产SQL Server数据库: 辅助服务器:用于存放主服务器上数据库的一个“镜像”数据库: 监控服务器:用来监控 ...

  6. cogs 1588. [USACO Feb04]距离咨询 倍增LCA

    1588. [USACO Feb04]距离咨询 ★★   输入文件:dquery.in   输出文件:dquery.out   简单对比时间限制:1 s   内存限制:256 MB [题目描述] 农夫 ...

  7. python爬虫——urllib使用代理

    收到粉丝私信说urllib库的教程还没写,好吧,urllib是python自带的库,没requests用着方便.本来嘛,python之禅(import this自己看)就说过,精简,效率,方便也是大家 ...

  8. Nginx配置文件模板

    主配置文件nginx.conf user nginx; #设置nginx服务的系统使用用户 worker_processes 1; #工作进程数(和cpu核心数保持一致) error_log /var ...

  9. 异数OS 织梦师-Xnign(四)-- 挑战100倍速Nginx,脚踩F5硬件负载均衡

    . 异数OS 织梦师-Xnign(四)– 挑战100倍速Nginx,脚踩F5硬件负载均衡 本文来自异数OS社区 github: https://github.com/yds086/HereticOS ...

  10. MySQL日志及索引

    MySQL物理结构: MySQL它是通过文件系统对数据进行储存和管理,从物理结构上分为日志文件和数据文件 日志文件: 日志文件记录了数据库操作的信息和一些错误信息,我们常用的日志文件有:错误日志.二进 ...