原题传送门

这题实际挺水的

先对两个字符串分别跑马拉车

就能求出1、2类扭动回文串最大的长度

考虑第三类的扭动回文串\(S(i,j,k)\),一定可以表示为\(A(i,l)+A(l+1,j)+B(j,k)\)或\(A(i,j)+B(j,l)+B(l+1,k)\),其中,第一段与第三段对称(第一段正着Hash和第三段反着Hash相同,数据水(某八位质数都不卡),单模数hash就行),第二段是一个回文子串,三段都可以是空串。

在A、B串上枚举扭动的回文串的中心mid,不难发现,以其在原串上能扩展出的最长回文子串为第二段进行扩展一定最优。对每个mid,以最长回文子串作为第二段,在A、B串上二分第一、三段的长度,其结果一定最优

代码(极其丑,常数极大)

#include <bits/stdc++.h>
#define getchar nc
#define N 200005
#define P 19260817
using namespace std;
inline char nc(){
static char buf[100000],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++;
}
inline char gc(){
char ch;
while((ch=getchar())<'A'||ch>'Z');
return ch;
}
inline int read()
{
register int x=0,f=1;register char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9')x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
return x*f;
}
inline void write(register int x)
{
if(!x)putchar('0');if(x<0)x=-x,putchar('-');
static int sta[20];register int tot=0;
while(x)sta[tot++]=x%10,x/=10;
while(tot)putchar(sta[--tot]+48);
}
inline int Min(register int a,register int b)
{
return a<b?a:b;
}
inline int Max(register int a,register int b)
{
return a>b?a:b;
}
int n,ans,p;
int a[N],b[N],sum[N],num[N],bin[N],f[N],g[N];
char ch;
inline bool check(register int l1,register int r1,register int l2,register int r2){
int x=0,y=0;
x=(sum[r1]-1ll*sum[l1-1]*bin[r1-l1+1]%P)%P;
y=(num[l2]-1ll*num[r2+1]*bin[r2-l2+1]%P)%P;
x=(x+P)%P,y=(y+P)%P;
return x==y?true:false;
}
inline int calc(register int j,register int k)
{
int l=0,r=Min(j,n-k+1),res=0;
while(l<=r)
{
int mid=l+r>>1;
if(check(j-mid+1,j,k,k+mid-1))
res=mid,l=mid+1;
else
r=mid-1;
}
return res;
}
int main()
{
n=read();
bin[0]=1;
for(register int i=1;i<=n;++i)
bin[i]=1ll*bin[i-1]*27%P;
a[0]=b[0]=0;
a[(n<<1)+2]=b[(n<<1)+2]=28;
a[1]=b[1]=27;
for(register int i=1;i<=n;++i)
ch=gc(),a[i<<1]=ch-'A'+1,a[i<<1|1]=27;
for(register int i=1;i<=n;++i)
ch=gc(),b[i<<1]=ch-'A'+1,b[i<<1|1]=27;
p=0;
for(register int i=2;i<=n<<1;++i)
{
if(i<=p+f[p])
f[i]=Min(f[(p<<1)-i],p+f[p]-i);
while(a[i-f[i]-1]==a[i+f[i]+1])
++f[i];
if(i+f[i]>p+f[p])
p=i;
}
p=0;
for(register int i=2;i<=n<<1;++i)
{
if(i<=p+g[p])
g[i]=Min(g[(p<<1)-i],p+g[p]-i);
while(b[i-g[i]-1]==b[i+g[i]+1])
++g[i];
if(i+g[i]>p+g[p])
p=i;
}
for(register int i=2;i<=n<<1;++i)
ans=Max(ans,Max(f[i],g[i]));
for(register int i=1;i<=n;++i)
sum[i]=(1ll*sum[i-1]*27%P+a[i<<1])%P;
for(register int i=n;i;--i)
num[i]=(1ll*num[i+1]*27%P+b[i<<1])%P;
int l=0,r=0;
for(register int i=2;i<=n<<1;++i)
{
l=(i-f[i]+1)>>1,r=(i+f[i])>>1;
ans=Max(ans,f[i]+(calc(l-1,r)<<1));
l=(i-g[i]+1)>>1,r=(i+g[i])>>1;
ans=Max(ans,g[i]+(calc(l,r+1)<<1));
}
write(ans);
return 0;
}

【题解】Luogu P4324 [JSOI2016]扭动的回文串的更多相关文章

  1. P4324 [JSOI2016]扭动的回文串

    传送门 对\(A\).\(B\)串各跑一遍\(manacher\),求出第\(1\).\(2\)类扭动回文串的最大长度. 考虑第三类的扭动回文串\(S(i,j,k)\),一定可以表示为\(A(i,l) ...

  2. 【BZOJ4755】 [Jsoi2016]扭动的回文串

    BZOJ4755 [Jsoi2016]扭动的回文串 Solution 考虑对于他给出的 A中的一个回文串: B中的一个回文串: 或者某一个回文的扭动字符串S(i,j,k) 这样子几个限制,我们1,2就 ...

  3. BZOJ4755: [JSOI2016]扭动的回文串——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4755 JYY有两个长度均为N的字符串A和B. 一个“扭动字符串S(i,j,k)由A中的第i个字符到 ...

  4. [BZOJ]4755: [Jsoi2016]扭动的回文串

    Time Limit: 10 Sec  Memory Limit: 512 MB Description JYY有两个长度均为N的字符串A和B. 一个"扭动字符串S(i,j,k)由A中的第i ...

  5. [bzoj4755][Jsoi2016]扭动的回文串

    来自FallDream的博客,未经允许,请勿转载,谢谢. JYY有两个长度均为N的字符串A和B. 一个“扭动字符串S(i,j,k)由A中的第i个字符到第j个字符组成的子串与B中的第j个字符到第k个字符 ...

  6. [JSOI2016]扭动的回文串

    题目 非常板子了 看到求什么最长的回文,我们就想到枚举回文中心的方法 首先对于这个回文串只包含在一个串内的情况,我们随便一搞就可以了,大概\(Manacher\)一下就没有了 对于那种扭动的回文串,我 ...

  7. BZOJ4755 [JSOI2016]扭动的回文串 【后缀数组】【manacher】

    题目分析: 我写了史上最丑的后缀数组,怎么办? 首先manacher一遍两个串,这样只用考虑第三问.用$作为间隔符拼接两个串,把第一个串翻转.枚举回文中心,取最长的回文串,对于剩下的部分利用LCP匹配 ...

  8. BZOJ4755 JSOI2016扭动的回文串(二分答案+哈希)

    显然答案应该是由单串以某位置为中心的极长回文串继续在另一个串里拓展得到的.枚举中间位置二分答案,哈希判断即可.注意考虑清楚怎么处理偶回文,比如像manacher一样加分隔符. #include< ...

  9. BZOJ4755:[JSOI2016]扭动的回文串

    浅谈\(Manacher\):https://www.cnblogs.com/AKMer/p/10431603.html 题目传送门:https://lydsy.com/JudgeOnline/pro ...

随机推荐

  1. C++学习 —— 住着魔鬼的细节

    13周的C++课程转眼就学完了5周,C++的标准基本上已经覆盖到了.再加上coding了上百行,总算是对C++有了一个基本的了解.接下来的学习会是关于STL的,所以在此对目前所学做一个小的总结. th ...

  2. Scala 中方法扩展实践

    前言 这个名字不知道取得是否合适,简单来说要干的事情就是给某个类型添加一些扩展方法,此场景在各种语言中都会用到,比如 C# 语言,如果我们使用一个别人写好的类库,而又想给某个类库添加一些自己封装的方法 ...

  3. python生成组织架构图(网络拓扑图、graph.editor拓扑图编辑器)

    Graph.Editor是一款基于HTML5技术的拓补图编辑器,采用jquery插件的形式,是Qunee图形组件的扩展项目,旨在提供可供扩展的拓扑图编辑工具, 拓扑图展示.编辑.导出.保存等功能,此外 ...

  4. R 544

    F2: 什么dfs根本不会啊,只会瞎贪心... 我们考虑先连哪些边,对于u,v两个点,如果u,v所在的联通块都与1相连的话,那我们肯定先不连这种边吧. 因为太亏了啊... 总而言之我们要在不影响答案的 ...

  5. Vs code常用插件

    Vs code常用插件 1.View In Browser 由于 VSCode 没有提供直接在浏览器中打开文件的内置界面,所以此插件在快捷菜单中添加了在默认浏览器查看文件选项,以及在客户端(Firef ...

  6. 将 Entity Framework、LINQ 和 Model-First 用于 Oracle 数据库

    目的 本教程介绍如何使用 Entity Framework.语言集成查询 (LINQ),以及如何对 Oracle 数据库使用 Model-First 以生成数据定义语言 (DDL) 脚本. 所需时间 ...

  7. openssl内核升级

    由于工作需要,防止安全漏洞需要对openssl升级现在整理出centos6.8和ubuntu14.4升级 centos升级openssl 1.首先去OpenSSL的网站 https://www.ope ...

  8. Javaweb实现对mongodb的增删改查(附带源代码)

    运行截图: 删除后的信息 项目源代码:https://www.cnblogs.com/post/readauth?url=/zyt-bg/p/9807396.html

  9. php 延迟静态绑定: static关键字

    abstract class DomainObject { public static function create() { return new self(); } } class User ex ...

  10. Gradle依赖无法下载问题

    之前一直在使用maven构建框架,对gradle还比较陌生.最近碰到依赖一直无法下载的问题.我采用了以下步骤进行排查: 1. 查看gradle本地仓库是否有依赖(肯定没有) 2. 查看私服是否有依赖( ...