一:背景

给定一个字符串,求出其最长回文子串。例如:

    1. s="abcd",最长回文长度为 1;
    2. s="ababa",最长回文长度为 5;
    3. s="abccb",最长回文长度为 4,即bccb。

以上问题的传统思路大概是,遍历每一个字符,以该字符为中心向两边查找。其时间复杂度为O(n^2),效率很差。

1975年,一个叫Manacher的人发明了一个算法,Manacher算法(中文名:马拉车算法),该算法可以把时间复杂度提升到O(n)。下面来看看马拉车算法是如何工作的。

二:算法过程分析

由于回文分为偶回文(比如 bccb)和奇回文(比如 bcacb),而在处理奇偶问题上会比较繁琐,所以这里我们使用一个技巧,具体做法是:在字符串首尾,及各字符间各插入一个字符(前提这个字符未出现在串里)。

举个例子:s="abbahopxpo",转换为s_new="$#a#b#b#a#h#o#p#x#p#o#"(这里的字符 $ 只是为了防止越界,下面代码会有说明),如此,s 里起初有一个偶回文abba和一个奇回文opxpo,被转换为#a#b#b#a##o#p#x#p#o#,长度都转换成了奇数

定义一个辅助数组int p[],其中p[i]表示以 i 为中心的最长回文的半径,例如:

i 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
s_new[i] $ # a # b # b # a # h # o # p # x # p #
p[i]   1 2 1 2 5 2 1 2 1 2 1 2 1 2 1 4 1 2 1

可以看出,p[i] - 1正好是原字符串中最长回文串的长度。

接下来的重点就是求解 p 数组,如下图:

设置两个变量,mx 和 id 。mx 代表以 id 为中心的最长回文的右边界,也就是mx = id + p[id]

假设我们现在求p[i],也就是以 i 为中心的最长回文半径,如果i < mx,如上图,那么:

if (i < mx)
p[i] = min(p[ * id - i], mx - i);

2 * id - i为 i 关于 id 的对称点,即上图的 j 点,而p[j]表示以 j 为中心的最长回文半径,因此我们可以利用p[j]来加快查找。

三:代码

//指定位置判断回文,此题为指定包含最后一个的最长回文序列。
char ma[maxn*], s[maxn];
int mp[maxn*];
int ans,Mlen;
void Manacher(char s[],int len)
{
int l=;
ma[l++]='$';
ma[l++]='#';
for(int i=; i<len; i++)
{
ma[l++]=s[i];
ma[l++]='#';
}
ma[l]=;
int mx=,id=;
for(int i=; i<l; i++)
{
mp[i]=mx>i?min(mp[*id-i],mx-i):;
while(ma[i+mp[i]]==ma[i-mp[i]])
mp[i]++;
if(i+mp[i]>mx)
{
mx=i+mp[i];
id=i;
}
// 这里可以check(ma[i])
ans=max(ans,mp[i]-);
if(mp[i]-+i==l-)
Mlen=max(Mlen,mp[i]-);
}
} int main()
{
int T;
cin>>T;
int kcase = ;
while(T--)
{
memset(ma,,sizeof(ma));
memset(mp,,sizeof(mp));
cin>>s;
int len=strlen(s);
ans=;
Mlen=;
Manacher(s,len);
if(ans == len)
printf("Case %d: %d\n", kcase++, ans);
else
printf("Case %d: %d\n", kcase++, len - Mlen + len);
}
}

四:题目

这个题目就是在原来的基础上添加了一个判断条件,看清楚在哪里添加。

//101350I - 2017 ACM Arabella Collegiate Programming Contest - Mirrored String II
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=;
char ma[maxn*];
int mp[maxn*];
char s[maxn];
int check(char zzz)
{
if(zzz=='A'||zzz=='H'||zzz=='I'||zzz=='M'||zzz=='O'||zzz=='#'||
zzz=='T'||zzz=='U'||zzz=='V'||zzz=='W'||zzz=='X'||zzz=='Y')
return ;
return ;
} void Manacher(char s[],int len)
{
int l=;
ma[l++]='$';
ma[l++]='#';
for(int i=; i<len; i++)
{
ma[l++]=s[i];
ma[l++]='#';
}
ma[l]=;
int mx=,id=;
for(int i=; i<l; i++)
{
mp[i]=mx>i?min(mp[*id-i],mx-i):;
while(check(ma[i+mp[i]])&&ma[i+mp[i]]==ma[i-mp[i]])//在这里添加check
{
mp[i]++;
}
if(i+mp[i]>mx)
{
mx=i+mp[i];
id=i;
}
}
} int main()
{
int T;
cin>>T;
while(T--)
{
cin>>s;
int len=strlen(s);
Manacher(s,len);
int ans=;
for(int i=; i<len*+; i++)
if(check(ma[i]))//这里添加check
ans=max(ans,mp[i]-);
cout<<ans<<endl;
}
}

manacher算法求最长回文子序列的更多相关文章

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

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

  2. Manacher算法——求最长回文子串

    首先,得先了解什么是回文串.回文串就是正反读起来就是一样的,如“abcdcba”.我们要是直接采用暴力方法来查找最长回文子串,时间复杂度为O(n^3),好一点的方法是枚举每一个字符,比较较它左右距离相 ...

  3. manacher算法求最长回文子串

    一:背景 给定一个字符串,求出其最长回文子串.例如: s="abcd",最长回文长度为 1: s="ababa",最长回文长度为 5: s="abcc ...

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

    1 概述(扯淡) 在了解Manacher算法之前,我们得先知道什么是回文串和子串. 回文串,就是正着看反着看都一样的字符串.比如说"abba"就是一个回文串,"abbc& ...

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

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

  6. leetcode 5 Longest Palindromic Substring(Manacher算法求最长回文串)

    应用一下manacher算法就可以O(n)求出结果了.可以参考hdu3068 substr(start,length)函数是这样用的: substr 方法 返回一个从指定位置开始,并具有指定长度的子字 ...

  7. Manacher算法求最长回文串模板

    #include <algorithm> #include <iostream> #include <cstring> #include <cstdio> ...

  8. NOIP2016提高组初赛(2)四、读程序写结果3、求最长回文子序列

    #include <iostream> using namespace std; int lps(string seq, int i, int j) { int len1, len2; i ...

  9. manacher 算法(最长回文串)

    manacher算法: 定义数组p[i]表示以i为中心的(包含i这个字符)回文串半径长 将字符串s从前扫到后for(int i=0;i<strlen(s);++i)来计算p[i],则最大的p[i ...

随机推荐

  1. 【CodeForces】679 B. Bear and Tower of Cubes

    [题目]B. Bear and Tower of Cubes [题意]有若干积木体积为1^3,2^3,...k^3,对于一个总体积X要求每次贪心地取<=X的最大积木拼上去(每个只能取一次)最后总 ...

  2. (值类型引用类型)和null的关系

    1.null    null表示变量没有指向任何对象. 2.值类型    包括 bool.结构体.枚举.int.double.float等等 .在.NET中值类型都继承自ValueType. 3. 引 ...

  3. POJ 1050 To the Max (最大子矩阵和)

    题目链接 题意:给定N*N的矩阵,求该矩阵中和最大的子矩阵的和. 题解:把二维转化成一维,算下就好了. #include <cstdio> #include <cstring> ...

  4. 【洛谷 P3965】 [TJOI2013]循环格(费用流)

    题目链接 回路限制经典题. 每个点拆成入点和出点,源点连每个点的出点,流量1,费用0,每个点出点连汇点,流量1,费用0,入点和出点之间没有边. 也就是说每个点必须靠其他点流来的流量来流入汇点,同时自己 ...

  5. 【leetcode 简单】第三十二题 买卖股票的最佳时机Ⅱ

    给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格. 设计一个算法来计算你所能获取的最大利润.你可以尽可能地完成更多的交易(多次买卖一支股票). 注意:你不能同时参与多笔交易(你必须在再次 ...

  6. flex图片布局

    <!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title>f ...

  7. 2017ACM暑期多校联合训练 - Team 2 1003 HDU 6047 Maximum Sequence (线段树)

    题目链接 Problem Description Steph is extremely obsessed with "sequence problems" that are usu ...

  8. HDU 2067 小兔的棋盘 (模拟)

    题目链接 Problem Description 小兔的叔叔从外面旅游回来给她带来了一个礼物,小兔高兴地跑回自己的房间,拆开一看是一个棋盘,小兔有所失望.不过没过几天发现了棋盘的好玩之处.从起点(0, ...

  9. low逼三人组、nb二人组、归并、希尔排序----小结

  10. Spring Boot企业级博客系统实战视频教程

    欢迎关注我的微信公众号:"Java面试通关手册" 回复关键字" springboot "免费领取(一个有温度的微信公众号,期待与你共同进步~~~坚持原创,分享美 ...