bzoj2342 [Shoi2011] 双倍回文

原博客地址

题目链接


因该算是一道挺不错的题,需要仔细思索一下

首先我可以看出:
(1)我们找到的串的本身也是一个回文串(显然)
(2)这个回文串的长度一定是偶数(显然)
(3)左右两个串一定也是偶数长度的回文串(显然)

那么我们先用manacher处理出以每个字符为中心的回文串长度
由于我们所需处理的这些串的长度都为偶数,所以这些串的中心都在manacher时的那些填充字符上(显然)

那么我们就先枚举大串的中心i,设左边小串的中心为j
那么j+rad[j]>=i   (rad[]为manacher中处理出的数组)
由于左边一定是回文串,那么rad[j]就应该要覆盖到i(不然怎么保证左边是回文串),而如果左边得到保证,那么右边也一定符合条件(对称)
所以我们就只需求出满足条件的最左侧的j

然后我们对j也有一个枚举范围,那就是在i的回文串范围内,并且还在i-rad[i]/2 ~ i 之间,不然不够

这样我们就可以初步得出一个枚举算法,那就是对于每个i,在一定范围内枚举j,找最优解
据说这个算法是可过的,但是复杂度。。。。似乎不是太乐观

于是需要优化
该优化其实也是显然的

如果我们曾枚举过一个j,它不能覆盖到当前枚举的i(也就是j+rad[j]
那么这个j,用一定不能覆盖到i+1(显然)
也就是说这个j在之后的计算中都没有用了,我们就不需要枚举了

这样我们就可以在枚举j的时候一段一段的跳,以降低复杂度
而实现这个过程,我们可以用并查集
每次都将没用的j的父亲指向j+1,然后跳到getfather(j+1)
这样就轻松完成了分段跳这个优化

最后在分析一下复杂度
(1)manacher  O(n)
(2)并查集    O(nα(n))
(3)每个点最多被删n次 O(n)
(4)每个点最多被利用一次 O(n)
(5)每个点最多被枚举一次 O(n)
这个复杂度真的是怎么算怎么舒心,而且代码很好实现 

#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
using namespace std;

char s[500005];
char str[1000005];
int rad[1000005];
int l,i,j,k,fa[1000005],ans;

int manacher(char s[],int l)
{
   int i,j,k,ans=0;
   for(i=1;i<=l;++i)str[i<<1]=s[i],str[(i<<1)+1]='#';
   str[1]='#';str[l*2+1]='#';str[0]='&';str[l*2+2]='$';
   l=l*2+1;j=0;
   for(i=1;i<=l;)
   {
       while(str[i-j-1]==str[i+j+1])++j;
       rad[i]=j;if(j>ans)ans=j;
       for(k=1;k<=j&&rad[i]-k!=rad[i-k];++k)rad[i+k]=min(rad[i-k],rad[i]-k);
       i+=k;j=max(j-k,0);
   }
   return ans;
}

int get(int p)
{
   if(fa[p]==p)return p;
   fa[p]=get(fa[p]);
   return fa[p];
}

int main()
{
   scanf("%d",&l);
   scanf("%s",s+1);
   manacher(s,l);
   l=l*2+1;
   for(i=1;i<=l;++i)
   if(str[i]=='#')fa[i]=i;else fa[i]=i+1;
   for(i=3;i<l;i+=2)
   {
       j=get(max(i-rad[i]/2,1));
       for(;j<i&&j+rad[j]<i;fa[j]=get(j+1),j=fa[j]);
       if(j<i)if((i-j)*2>ans)ans=(i-j)*2;
   }
   printf("%d\n",ans);
   system("pause");
}



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值