JSOI 2016 扭动的字符串

题面描述

给出两个长度为\(n\)的字符串\(A,B\)

\(S(i,j,k)\)表示把\(A\)中的\([i,j]\)和\(B\)中的\([j,k]\)拼接起来的字符串

问所有回文的\(S(i,j,k)\)或者\(A,B\)中的回文子串的最长长度

思路

枚举回文串的中心。

可以发现,如果能在当前字符串内扩展就尽量扩展,不能扩展了再尝试和另一个字符串匹配。

对于前者,使用\(manacher\)算法

对于后者,二分一个长度,用\(hash\)判断能否匹配。

以上

代码

#include<bits/stdc++.h>
#define pii pair<int,int>
#define mkp(x,y) make_pair(x,y)
using namespace std;
const int sz=2e5+7;
const int p1=998244353;
const int p2=1e8+7;
const int q1=5271314;
const int q2=2374899;
int n,m;
int ans;
int p[2][sz];
char s[2][sz];
char ss[2][sz<<1];
int w[2][sz];
int pre[2][2][sz],suf[2][2][sz];//字符串,哈希,位置
void init(int tp){
int mid=0,r=0;
for(int i=1;i<=m;i++){
if(i<r) p[tp][i]=min(r-i,p[tp][2*mid-i]);
else p[tp][i]=1;
while(ss[tp][i+p[tp][i]]==ss[tp][i-p[tp][i]]) p[tp][i]++;
if(i+p[tp][i]>r) mid=i,r=i+p[tp][i];
}
}
void hash(int tp){
int hs0,hs1;
hs0=hs1=0;
for(int i=1;i<=n;i++){
hs0=(1ll*hs0*q1%p1+s[tp][i])%p1;
hs1=(1ll*hs1*q2%p2+s[tp][i])%p2;
pre[tp][0][i]=hs0;
pre[tp][1][i]=hs1;
}
hs0=hs1=0;
for(int i=n;i>=1;i--){
hs0=(1ll*hs0*q1%p1+s[tp][i])%p1;
hs1=(1ll*hs1*q2%p2+s[tp][i])%p2;
suf[tp][0][i]=hs0;
suf[tp][1][i]=hs1;
}
}
pii gs(int tp,int st,int l){
int sum1=pre[tp][0][st]-1ll*w[0][l]*pre[tp][0][st-l]%p1;
int sum2=pre[tp][1][st]-1ll*w[1][l]*pre[tp][1][st-l]%p2;
if(sum1<0) sum1+=p1;
if(sum2<0) sum2+=p2;
return mkp(sum1,sum2);
}
pii gn(int tp,int st,int l){
int sum1=suf[tp][0][st]-1ll*w[0][l]*suf[tp][0][st+l]%p1;
int sum2=suf[tp][1][st]-1ll*w[1][l]*suf[tp][1][st+l]%p2;
if(sum1<0) sum1+=p1;
if(sum2<0) sum2+=p2;
return mkp(sum1,sum2);
}
int main(){
scanf("%d",&n);
scanf("%s",s[0]+1);
scanf("%s",s[1]+1);
ss[0][0]=ss[1][0]='#';
ss[0][1]=ss[1][1]='|';
for(int i=1;i<=n;i++){
ss[0][2*i]=s[0][i];
ss[1][2*i]=s[1][i];
ss[0][2*i+1]=ss[1][2*i+1]='|';
}
m=2*n+1;
w[0][0]=w[1][0]=1;
for(int i=1;i<=n;i++){
w[0][i]=1ll*w[0][i-1]*q1%p1;
w[1][i]=1ll*w[1][i-1]*q2%p2;
}
init(0),init(1);
hash(0),hash(1);
int L,R,sl,sr,len,l,r,mid;
for(int i=2;i<m;i++){
L=i-p[0][i],R=i+p[0][i];
sl=L/2,sr=R/2-1;
len=p[0][i]-1;
l=0,r=min(sl,n-sr+1);
while(l<r){
mid=(l+r+1)>>1;
if(gs(0,sl,mid)==gn(1,sr,mid)) l=mid;
else r=mid-1;
}
ans=max(ans,len+2*l);
}
for(int i=2;i<m;i++){
L=i-p[1][i],R=i+p[1][i];
sl=L/2+1,sr=R/2;
len=p[1][i]-1;
l=0,r=min(sl,n-sr+1);
while(l<r){
mid=(l+r+1)>>1;
if(gs(0,sl,mid)==gn(1,sr,mid)) l=mid;
else r=mid-1;
}
ans=max(ans,len+2*l);
}
printf("%d\n",ans);
}

JSOI 2016 扭动的字符串的更多相关文章

  1. [JSOI 2016] 最佳团体(树形背包+01分数规划)

    4753: [Jsoi2016]最佳团体 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2003  Solved: 790[Submit][Statu ...

  2. 解题:JSOI 2016 最佳团体

    题面 0/1分数规划+树形背包检查 要求$\frac{\sum P_i}{\sum S_i}的最大值,$按照0/1分数规划的做法,二分一个mid之后把式子化成$\sum P_i=\sum S_i*mi ...

  3. [JSOI 2016] 灯塔

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=4850 [算法] 首先对不等式进行移项 : hj <= hi + p - sqr ...

  4. [JSOI 2016] 最佳团体

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=4753 [算法] 很明显的分数规划 可以用树形动态规划(树形背包)检验答案 时间复杂度 ...

  5. JSOI 2016 病毒感染 辅助Dp问题

    原题链接:https://www.luogu.com.cn/problem/P5774 分析 直接看这道题,第一个困惑点,那个绝对值的比较是什么东西,根据数学知识,我们可以知道这个意思是k到i的距离小 ...

  6. 《JAVASCRIPT高级程序设计》JSON语法/解析/序列化

    JSON是一种数据格式,不是一种编程语言. 一.语法 JSON语法可以表示以下三种类型的值:简单值.对象.数组. 1.简单值 最简单的JSON数据值就是简单值: 5 "hello world ...

  7. Chrome 调试技巧

    Chrome 调试技巧 1.alert 这个不用多说了,不言自明. 可参考:https://www.cnblogs.com/Michelle20180227/p/9110028.html 2.cons ...

  8. Python连接Access数据库

    前言 今天想要用Python访问Access数据库,折腾了半天,特记录一下 背景 最近想将一些文件记录下来,存入数据库,为此拿LabVIEW写了一个版本,记录环境配置为: LabVIWE:2015 A ...

  9. JNI_Z_08_创建Java对象

    1.步骤 : (1).获取 jclass (2).获取 构造函数的 method id (方法的名称始终为"<init>") (3).创建Java对象的两种方式: (3 ...

随机推荐

  1. Windows 隐藏控制台

    #pragma comment( linker, "/subsystem:\"windows\" /entry:\"mainCRTStartup\"& ...

  2. 线性dp——cf1096D

    dp[i][j]表示到第i位,与hard的匹配状态到达了第j位 每位有两种决策:消或者不消 分别转移一下即可 转移代码 ;i<n;++i) ;j<=;++j) { cmin(f[i+][j ...

  3. VS2010-MFC(常用控件:标签控件Tab Control 上)

    转自:http://www.jizhuomi.com/software/205.html 前面两节讲了树形控件Tree Control,本节开始讲解标签控件Tab Control,也可以称为选项卡控件 ...

  4. node.js在ubuntu上和windows上的安装

    Ubuntu 上安装 Node.js Node.js 源码安装 以下部分我们将介绍在Ubuntu Linux下安装 Node.js . 其他的Linux系统,如Centos等类似如下安装步骤. 在 G ...

  5. ajax跨域请求使用代理

    public class ProxyHandler extends HttpServlet {    private static final long serialVersionUID = 1L;  ...

  6. Vue的指令和成员

    目录 Vue的指令和成员 指令 表单 斗篷 条件 循环 成员 计算成员 监听成员 Vue的指令和成员 指令 表单 表单指令一般是和属性指令联合使用的,最常见的就是v-model="变量&qu ...

  7. springcloud系列12 config的使用

    config组件分为server端和client端 config的原理: 就是当我们将配置文件放置在git上面,那么configserver就会去拉取相关配置文件至本地: 可以看到我本地是拉去了配置文 ...

  8. java笔试之自守数

    自守数是指一个数的平方的尾数等于该数自身的自然数.例如:25^2 = 625,76^2 = 5776,9376^2 = 87909376.请求出n以内的自守数的个数 接口说明 /*功能: 求出n以内的 ...

  9. Luogu P2458 [SDOI2006]保安站岗(树形dp)

    P2458 [SDOI2006]保安站岗 题意 题目描述 五一来临,某地下超市为了便于疏通和指挥密集的人员和车辆,以免造成超市内的混乱和拥挤,准备临时从外单位调用部分保安来维持交通秩序. 已知整个地下 ...

  10. php访问其他网站接口

    使用函数:  file_get_contents($url); 传入接口url及其参数:如 $url="http://192.168.1.1/test.jsp?id=1&type=2 ...