最长双回文串——manacehr

题目

【题目描述】

顺序和逆序读起来完全一样的串叫做回文串。比如 acbca 是回文串,而 abc 不是(abc 的顺序为 “abc”,逆序为 “cba”,不相同)。
输入长度为 n 的串 S,求 S 的最长双回文子串 T, 即可将 T 分为两部分 X,Y,(|X|,|Y|≥1)且 X 和 Y 都是回文串。

【输入格式】

一行由小写英文字母组成的字符串 S。

【输出格式】

一行一个整数,表示最长双回文子串的长度。

【样例输入】

baacaabbacabb

【样例输出】

12

【数据范围与提示】

对于 10% 的数据,2≤|S|≤103。
对于 30% 的数据,2≤|S|≤104。
对于 100% 的数据,2≤|S|≤105。

题解

跑一遍 manacehr,然后分别 dp 存下以 $ i $ 为对称中心时的回文串长度,记录在左端点和右端点上,取两端回文串最大值即可

代码

 1 #include<bits/stdc++.h>
 2 #define LL long long
 3 #define _(d) while(d(isdigit(ch=getchar())))
 4 using namespace std;
 5 int R(){
 6     int x;bool f=1;char ch;_(!)if(ch=='-')f=0;x=ch^48;
 7     _()x=(x<<3)+(x<<1)+(ch^48);return f?x:-x;}
 8 const int N=3e5+5;
 9 int n,len,p[N],mx,id,f[N],g[N],ans;
10 char ch[N],s[N];
11 void manacher(){
12     len=strlen(ch+1);
13     for(int i=1;i<=len;i++)
14         s[++n]='#',s[++n]=ch[i];
15     s[0]='(',s[++n]='#',s[n+1]=')';
16     mx=1,id=1;
17     for(int i=1;i<=n;i++){
18         if(mx>i)p[i]=min(p[id*2-i],mx-i);
19         else p[i]=1;
20         while(s[i-p[i]]==s[i+p[i]])p[i]++;
21         if(i+p[i]>mx)mx=i+p[i],id=i;
22     }
23 }
24 int main(){
25     scanf("%s",ch+1);
26     manacher();
27     mx=1;
28     for(int i=1;i<=n;i++)
29         if(i+p[i]>mx){
30             for(int j=mx+1;j<=i+p[i];j++)
31                 f[j]=j-i+1;
32             mx=i+p[i]-1;
33         }
34     mx=n;
35     for(int i=n;i;i--)
36         if(i-p[i]<mx){
37             for(int j=mx-1;j>=i-p[i];j--)
38                 g[j]=i-j+1;
39             mx=i-p[i]+1;
40         }    
41     for(int i=1;i<=n;i++)
42         if(s[i]!='#')
43             ans=max(f[i]+g[i+2],ans);
44     cout<<ans<<endl;
45     return 0;
46 }
View Code

 

posted @ 2019-04-06 15:18  Chm_wt  阅读(139)  评论(0编辑  收藏  举报