博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[Shoi2011]双倍回文
阅读量:5818 次
发布时间:2019-06-18

本文共 1563 字,大约阅读时间需要 5 分钟。

Description

Description

Input

输入分为两行,第一行为一个整数,表示字符串的长度,第二行有个连续的小写的英文字符,表示字符串的内容。

Output

输出文件只有一行,即:输入数据中字符串的最长双倍回文子串的长度,如果双倍回文子串不存在,则输出0。

Sample Input

16
ggabaabaabaaball

Sample Output

12

HINT

N<=500000

Source

思路

显然每个双倍回文串被扩充(插入字符)后的结果一定是$A$AR$A$AR$,那么以第3个$为中心的整个串一定是回文串,以第2个$为中心的串的前半部分和以第4个$为中心的串的后半部分串也是回文串。那么可以依据这两个性质进行计算了:
1. 枚举中心点i
2. j为回文串能够包含于i为中心的回文串的最小节点序号。
3. 如果以j为中心的回文串可以到达i,那么更新ans值;否则删除j,并将j赋值为j右侧可以到达的第一个没有被删除的点,然后重复这个步骤。
但是上面这个步骤显然是不能过的,时间复杂度为O(n2),这个时候显然可以用并查集来优化了。

代码

#include 
#include
const int maxn=500000;char s[maxn+10],a[(maxn<<1)+10];int p[(maxn<<1)+10],id,rmax,k,n,ans;int fa[(maxn<<1)+10];int find(int x){ if(fa[x]) { return fa[x]=find(fa[x]); } else { return x; }}int main(){ scanf("%d%s",&n,s+1); a[0]='!'; a[1]='$'; for(register int i=1; i<=n; ++i) { a[i<<1]=s[i]; a[i<<1|1]='$'; } n=n<<1|1; a[n+1]='*'; id=rmax=p[1]=1; for(register int i=2; i<=n; ++i) { if(i>rmax) { p[i]=1; } else { if(p[(id<<1)-i]
rmax) { rmax=i+p[i]-1; id=i; } } for(register int i=1; i<=n; i+=2)//寻找中心点 { int j;//j代表左半部分的中心点 if(i-(p[i]>>1)>1) { j=i-(p[i]>>1); } else { j=1; } j=find(j);//最右侧没有被删除的节点 while((j
ans) { ans=(i-j)<<1;//更新ans } } } printf("%d\n",ans); return 0;}

转载于:https://www.cnblogs.com/Canopus-wym/p/10376283.html

你可能感兴趣的文章
re:Invent解读:没想到你是这样的AWS
查看>>
PyTips 0x02 - Python 中的函数式编程
查看>>
阿里云安全肖力:安全基础建设是企业数字化转型的基石 ...
查看>>
使用《Deep Image Prior》来做图像复原
查看>>
Linux基础命令---rmdir
查看>>
iOS sqlite3(数据库)
查看>>
粤出"飞龙",打造新制造广东样本
查看>>
编玩边学获数千万元A轮融资,投资方为君联资本
查看>>
蓝图(Blueprint)详解
查看>>
Spark之SQL解析(源码阅读十)
查看>>
Android图片添加水印图片并把图片保存到文件存储
查看>>
比特币系统采用的公钥密码学方案和ECDSA签名算法介绍——第二部分:代码实现(C语言)...
查看>>
海贼王十大悲催人物
查看>>
BigDecimal 舍入模式(Rounding mode)介绍
查看>>
开源 免费 java CMS - FreeCMS1.2-标签 infoSign
查看>>
开源 免费 java CMS - FreeCMS1.9 移动APP生成栏目列表数据
查看>>
虚拟机新增加硬盘,不用重启读到新加的硬盘
查看>>
Java IO流详尽解析
查看>>
邮件服务系列之四基于虚拟用户的虚拟域的邮件系统(安装courier-authlib以及部分配置方法)...
查看>>
Linux VSFTP服务器
查看>>