BZOJ3654 : 图样图森破
考虑枚举回文中心,然后向两边扩展,当匹配到当前串的边界的时候,枚举下一个串接上。
这个过程可以通过记忆化搜索来完成,设:
$f[i][0]$表示对于$i$这个位置,$[i,串结尾]$等待匹配的最长回文子串。
$f[i][1]$表示对于$i$这个位置,$[串开头,i]$等待匹配的最长回文子串。
如果在转移的过程中发现两个串都已经匹配到了边界,或者转移有环,那么说明答案无限。
用后缀数组支持lcp的询问,时间复杂度$O(nL+L\log L)$。
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=400010,M=105;
int n,m,i,j,k,x,y,st[M],en[M],from[N],f[N][2],v[N][2],vis[N][2],ans;char a[N],s[N];
namespace SA{
int n,rk[N],sa[N],height[N],tmp[N],cnt[N],Log[N],f[18][N];char s[N];
void build(int n,int m){
int i,j,k;n++;
for(i=0;i<n;i++)cnt[rk[i]=s[i]]++;
for(i=1;i<m;i++)cnt[i]+=cnt[i-1];
for(i=0;i<n;i++)sa[--cnt[rk[i]]]=i;
for(k=1;k<=n;k<<=1){
for(i=0;i<n;i++){
j=sa[i]-k;
if(j<0)j+=n;
tmp[cnt[rk[j]]++]=j;
}
sa[tmp[cnt[0]=0]]=j=0;
for(i=1;i<n;i++){
if(rk[tmp[i]]!=rk[tmp[i-1]]||rk[tmp[i]+k]!=rk[tmp[i-1]+k])cnt[++j]=i;
sa[tmp[i]]=j;
}
memcpy(rk,sa,n*sizeof(int));
memcpy(sa,tmp,n*sizeof(int));
if(j>=n-1)break;
}
for(j=rk[height[i=k=0]=0];i<n-1;i++,k++)
while(~k&&s[i]!=s[sa[j-1]+k])height[j]=k--,j=rk[sa[j]+1];
for(i=2;i<n;i++)Log[i]=Log[i>>1]+1;
for(i=1;i<n;i++)f[0][i]=height[i];
for(j=1;j<18;j++)for(i=1;i+(1<<j)-1<n;i++)f[j][i]=min(f[j-1][i],f[j-1][i+(1<<(j-1))]);
}
inline int ask(int x,int y){
int k=Log[y-x+1];
return min(f[k][x],f[k][y-(1<<k)+1]);
}
inline int lcp(int x,int y){
if(x==y)return N;
x=rk[x],y=rk[y];
if(x>y)swap(x,y);
return ask(x+1,y);
}
}
inline int query(int x,int y){
return min(SA::lcp(x,m-1-y),min(en[from[x]]-x,y-st[from[y]])+1);
}
void getinf(){
puts("Infinity");
exit(0);
}
int dfs(int u,int p){
if(vis[u][p])getinf();
if(v[u][p])return f[u][p];
v[u][p]=vis[u][p]=1;
int&ret=f[u][p];
if(!p){
for(int i=1;i<=n;i++){
int k=query(u,en[i]);
int x=u+k-1,y=en[i]-k+1;
if(x<en[from[u]]&&y>st[i])ret=max(ret,k*2);
else if(x==en[from[u]]&&y==st[i])getinf();
else if(x==en[from[u]])ret=max(ret,k*2+dfs(y-1,1));
else ret=max(ret,k*2+dfs(x+1,0));
}
}else{
for(int i=1;i<=n;i++){
int k=query(st[i],u);
int x=u-k+1,y=st[i]+k-1;
if(x>st[from[u]]&&y<en[i])ret=max(ret,k*2);
else if(x==st[from[u]]&&y==en[i])getinf();
else if(x==st[from[u]])ret=max(ret,k*2+dfs(y+1,0));
else ret=max(ret,k*2+dfs(x-1,1));
}
}
vis[u][p]=0;
return ret;
}
int main(){
scanf("%d",&n);
for(i=1;i<=n;i++){
scanf("%s",a);
st[i]=m;
for(j=0;a[j];j++)from[m]=i,s[m++]=a[j];
en[i]=m-1;
}
m<<=1;
for(i=0,j=m-1;i<j;i++,j--)s[j]=s[i],from[j]=from[i];
for(SA::n=m,i=0;i<m;i++)SA::s[i]=s[i];
SA::build(m,128);
for(i=1;i<=n;i++)ans=max(ans,max(dfs(st[i],0),dfs(en[i],1)));
for(i=1;i<=n;i++){
for(j=st[i];j<=en[i];j++){
k=query(j,j);
x=j-k+1,y=j+k-1;
if(x>st[i]&&y<en[i])ans=max(ans,k*2-1);
else if(x==st[i]&&y==en[i])getinf();
else if(x==st[i])ans=max(ans,k*2-1+dfs(y+1,0));
else ans=max(ans,k*2-1+dfs(x-1,1));
}
for(j=st[i];j<en[i];j++){
k=query(j+1,j);
x=j-k+1,y=j+k;
if(x>st[i]&&y<en[i])ans=max(ans,k*2);
else if(x==st[i]&&y==en[i])getinf();
else if(x==st[i])ans=max(ans,k*2+dfs(y+1,0));
else ans=max(ans,k*2+dfs(x-1,1));
}
}
return printf("%d",ans),0;
}
BZOJ3654 : 图样图森破的更多相关文章
- P3900 [湖南集训]图样图森破
P3900 [湖南集训]图样图森破 链接 分析: 感觉像个暴力. 可以枚举回文串的回文中心,即枚举一个串,枚举一个串的位置作为回文中心,然后求出这个串内的回文串的长度. 此时如果回文串两端都没有到这个 ...
- Bzoj 3654 图样图森波 题解
3654: 图样图森破 Time Limit: 30 Sec Memory Limit: 512 MBSubmit: 123 Solved: 66[Submit][Status][Discuss] ...
- ES6 箭头函数中的 this?你可能想多了(翻译)
箭头函数=>无疑是ES6中最受关注的一个新特性了,通过它可以简写 function 函数表达式,你也可以在各种提及箭头函数的地方看到这样的观点——“=> 就是一个新的 function”. ...
- Windbg Extension NetExt 使用指南 【2】 ---- NetExt 的基本命令介绍
摘要 : 本章节介绍NetExt常用的命令. 并且对SOS进行一些对比. NetExt的帮助 要想玩好NetExt, 入门就得看帮助. 看NetExt的帮助可以调用!whelp 命令. 这样hi列举出 ...
- 自己动手之使用反射和泛型,动态读取XML创建类实例并赋值
前言: 最近小匹夫参与的游戏项目到了需要读取数据的阶段了,那么觉得自己业余时间也该实践下数据相关的内容.那么从哪入手呢?因为用的是Unity3d的游戏引擎,思来想去就选择了C#读取XML文件这个小功能 ...
- Hadoop学习之旅三:MapReduce
MapReduce编程模型 在Google的一篇重要的论文MapReduce: Simplified Data Processing on Large Clusters中提到,Google公司有大量的 ...
- [LeetCode] Maximum Size Subarray Sum Equals k 最大子数组之和为k
Given an array nums and a target value k, find the maximum length of a subarray that sums to k. If t ...
- [LeetCode] Pow(x, n) 求x的n次方
Implement pow(x, n). 这道题让我们求x的n次方,如果我们只是简单的用个for循环让x乘以自己n次的话,未免也把LeetCode上的想的太简单了,一句话形容图样图森破啊.OJ因超时无 ...
- [LeetCode] Median of Two Sorted Arrays 两个有序数组的中位数
There are two sorted arrays nums1 and nums2 of size m and n respectively. Find the median of the two ...
随机推荐
- LNMP平台搭建---Nginx安装篇
在上一篇博文<LNMP平台搭建---Linux系统安装篇>中,我们安装了CentOS版本的Linux操作系统,现在,我们来安装一个Web服务器,大标题写着LNMP,其中的N就是Nginx, ...
- SQL跨项目查询语法
EXEC sp_addlinkedserver 'ITSV', '', 'SQLOLEDB', '192.168.1.248' EXEC sp_addlinkedsrvlogin 'ITSV', 'f ...
- WIN10 新建ORACLE实例
1 管理员身份进入CMD环境,执行DBCA命令,在弹出窗口的引导中,完成实例创建 2 如果在创建过程中没有选择适当的字符集(最好采用默认字符集),如下图所示,在进入PLSQL DEVELOPER的时候 ...
- Linux 端口-> PID -> 启动目录
1. lsof -i :8443 找到PID 比如说是5413 2. ps aux | grep 5413 可以得到一些信息 3. 除了第二步的方式,更直观的是 cd /pro ...
- subversion安装使用
这里仅针对subversion进行说明,未完待续. 一.下载subversion并安装: a.subversion 要做svn服务器的必须装 b.Tortoisesvn 仅仅是访问svn服务器的客户端 ...
- [Tips] JavaScript 使用hash 对象传参
转自Web 前端开发修炼之道. 在JavaScript 中funciton 包含多个参数的时候,我们想要实现可选参数的功能,传很多个null 其实是个很讨厌的事情,这个时候就可以使用这个技巧. 具体见 ...
- SQL常见错误及处理方法
1.情况:数据库引擎安装失败,报类似权限不足的错误 解决:可能由于计算机名和用户名相同导致,更改计算机名,卸载干净重装即可
- linux c学习笔记----互斥锁属性
转自:http://lobert.iteye.com/blog/1762844 互斥锁属性 使用互斥锁(互斥)可以使线程按顺序执行.通常,互斥锁通过确保一次只有一个线程执行代码的临界段来同步多个线程. ...
- Java中的wait和sleep
sleep()和wait() 首先,Java中的多线程是一种抢占式的机制,而不是分时机制.抢占式的机制是有多个线程处于可运行状态,但是只有一个线程在运行. 这种机制决定了,对于同一对象的多线程访问,必 ...
- AngularJS - 指令入门
指令,我将其理解为AngularJS操作HTML element的一种途径. 由于学习AngularJS的第一步就是写内置指令ng-app以指出该节点是应用的根节点,所以指令早已不陌生. 这篇日志简单 ...