『题解』[NOI2016]优秀的拆分
如果一个字符串可以被拆分为\(AABB\)的形式,其中$A和 B是任意非空字符串,则我们称该字符串的这种拆分是优秀的。
例如,对于字符串\(aabaabaa\),如果令\(A=aab\),\(B=a\),我们就找到了这个字符串拆分成\(AABB\)的一种方式。
一个字符串可能没有优秀的拆分,也可能存在不止一种优秀的拆分。比如我们令\(A=a\),\(B=baa\),也可以用 AABB表示出上述字符串;但是,字符串\(abaabaa\)就没有优秀的拆分。
现在给出一个长度为\(n\)的字符串\(S\),我们需要求出,在它所有子串的所有拆分方式中,优秀拆分的总个数。这里的子串是指字符串中连续的一段。
以下事项需要注意:
出现在不同位置的相同子串,我们认为是不同的子串,它们的优秀拆分均会被记入答案。
在一个拆分中,允许出现\(A=B\)。例如\(cccc\)存在拆分\(A=B=c\)。
字符串本身也是它的一个子串。
对于\(AABB\),我们完全可以只考虑\(AA\),这样令\(f[i]\)表示以i结尾的\(AA\)数量,\(g[i]\)表示以\(i\)开头的\(AA\)数量,那么显然就是\(sigma(f[i]g[i+1])\)。
对于\(AA\)怎么求,大体的思路和URAL1297:Palindrome求回文串是一样的,就是通过比较后缀的公共前缀来得到AA的长度,进而求出这段区间内\(f[i]g[i]\)的值。
但是这样显然是\(O(n^{2})\)的。
我们用分块的思想,枚举\(l\),将字符串分成\(l\)大小的块,则长度为\(l\)的\(AA\)一定最少跨过两个块,于是对于块边界点,求一次公共前缀和后缀,拼在一起就是我们所要的答案,复杂度调和级数\(O(n×log_{2}^{n})\)。
注意,为了让复杂度正确,我们对区间的\(f\)和\(g\)差分。
代码:
#include<cstdio>
#include<cmath>
#include<iostream>
#include<vector>
#include<cstring>
#include<algorithm>
#include<cctype>
using namespace std;
typedef long long ll;
const int N=2e6+10;
char s[N];
int n,m,rk[N],height[N],sa[N],w[N],cas,dp[N][21],lg[N];
int f[N],g[N];
inline int qpow(int a)
{
return 1<<a;
}
inline bool pan(int *x,int i,int j,int k)
{
int ti=i+k<n?x[i+k]:-1;
int tj=j+k<n?x[j+k]:-1;
return ti==tj&&x[i]==x[j];
}
void SA_init()
{
int *x=rk,*y=height,r=256;
for(int i=0; i<r; i++)w[i]=0;
for(int i=0; i<n; i++)w[s[i]]++;
for(int i=1; i<r; i++)w[i]+=w[i-1];
for(int i=n-1; i>=0; i--)sa[--w[s[i]]]=i;
r=1;
x[sa[0]]=0;
for(int i=1; i<n; i++)
x[sa[i]]=s[sa[i]]==s[sa[i-1]]?r-1:r++;
for(int k=1; r<n; k<<=1)
{
int yn=0;
for(int i=n-k; i<n; i++)y[yn++]=i;
for(int i=0; i<n; i++)
if(sa[i]>=k)y[yn++]=sa[i]-k;
for(int i=0; i<r; i++)w[i]=0;
for(int i=0; i<n; i++)w[x[y[i]]]++;
for(int i=1; i<r; i++)w[i]+=w[i-1];
for(int i=n-1; i>=0; i--)sa[--w[x[y[i]]]]=y[i];
swap(x,y);
r=1;
x[sa[0]]=0;
for(int i=1; i<n; i++)
x[sa[i]]=pan(y,sa[i],sa[i-1],k)?r-1:r++;
}
}
inline void height_init()
{
int i,j,k=0;
for(int i=1; i<=n; i++)rk[sa[i]]=i;
for(int i=0; i<n; i++)
{
if(k)k--;
j=sa[rk[i]-1];
while(s[i+k]==s[j+k])k++;
height[rk[i]]=k;
}
}
void st_init()
{
for(int i=1; i<=n; i++)
{
dp[i-1][0]=height[i];
lg[i]=lg[i-1];
if((1<<lg[i]+1)==i)lg[i]++;
}
for(int j=1; j<=lg[n]; j++)
{
for(int i=0; i<n; i++)
{
if(i+qpow(j)-1>=n)break;
dp[i][j]=min(dp[i][j-1],dp[i+qpow(j-1)][j-1]);
}
}
}
int lcp(int a,int b)
{
int l=rk[a],r=rk[b];
if(r<l)swap(l,r);
l--;
r--;
if(r<0)return 0;
l++;
int len=r-l+1;
int k=lg[len];
int h=qpow(k);
return min(dp[l][k],dp[r-h+1][k]);
}
int main()
{
scanf("%d",&cas);
while(cas--)
{
memset(f,0,sizeof(f));
memset(g,0,sizeof(g));
cin>>s;
m=strlen(s),n=2*m+1;
s[m]='$';
for(int i=m+1; i<n; i++)
{
s[i]=s[n-i-1];
}
s[n++]=0;
SA_init();
n--;
height_init();
st_init();
for(int l=1; l<=m/2; l++)
{
for(int i=0,j=l; j<m; i+=l,j+=l)
{
int p=min(l,lcp(i,j));
int s=min(l,lcp(n-i-1,n-j-1));
if(p+s-1>=l)
{
f[j-s+l]++;
f[j+p]--;
g[i-s+1]++;
g[i+p-l+1]--;
}
}
}
ll ans=0;
for(int i=1; i<m; i++)
{
f[i]+=f[i-1];
g[i]+=g[i-1];
}
for(int i=0; i<m-1; i++)
{
ans+=(ll)f[i]*g[i+1];
}
printf("%lld\n",ans);
}
return 0;
}
『题解』[NOI2016]优秀的拆分的更多相关文章
- 题解-NOI2016 优秀的拆分
NOI2016 优秀的拆分 \(T\) 组测试数据.求字符串 \(s\) 的所有子串拆成 \(AABB\) 形式的方案总和. 数据范围:\(1\le T\le 10\),\(1\le n\le 3\c ...
- [NOI2016]优秀的拆分&&BZOJ2119股市的预测
[NOI2016]优秀的拆分 https://www.lydsy.com/JudgeOnline/problem.php?id=4650 题解 如果我们能够统计出一个数组a,一个数组b,a[i]表示以 ...
- 【BZOJ4560】[NOI2016]优秀的拆分
[BZOJ4560][NOI2016]优秀的拆分 题面 bzoj 洛谷 题解 考虑一个形如\(AABB\)的串是由两个形如\(AA\)的串拼起来的 那么我们设 \(f[i]\):以位置\(i\)为结尾 ...
- [UOJ#219][BZOJ4650][Noi2016]优秀的拆分
[UOJ#219][BZOJ4650][Noi2016]优秀的拆分 试题描述 如果一个字符串可以被拆分为 AABBAABB 的形式,其中 A 和 B 是任意非空字符串,则我们称该字符串的这种拆分是优秀 ...
- [NOI2016]优秀的拆分(SA数组)
[NOI2016]优秀的拆分 题目描述 如果一个字符串可以被拆分为 \(AABB\) 的形式,其中 A和 B是任意非空字符串,则我们称该字符串的这种拆分是优秀的. 例如,对于字符串 \(aabaaba ...
- luogu1117 [NOI2016]优秀的拆分
luogu1117 [NOI2016]优秀的拆分 https://www.luogu.org/problemnew/show/P1117 后缀数组我忘了. 此题哈希可解决95分(= =) 设\(l_i ...
- 『题解』洛谷P1063 能量项链
原文地址 Problem Portal Portal1:Luogu Portal2:LibreOJ Portal3:Vijos Description 在\(Mars\)星球上,每个\(Mars\)人 ...
- BZOJ4650:[NOI2016]优秀的拆分——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=4650 https://www.luogu.org/problemnew/show/P1117 如果 ...
- [NOI2016]优秀的拆分 后缀数组
题面:洛谷 题解: 因为对于原串的每个长度不一定等于len的拆分而言,如果合法,它将只会被对应的子串统计贡献. 所以子串这个限制相当于是没有的. 所以我们只需要对于每个位置i求出f[i]表示以i为开头 ...
随机推荐
- Spring Boot 2.X(七):Spring Cache 使用
Spring Cache 简介 在 Spring 3.1 中引入了多 Cache 的支持,在 spring-context 包中定义了org.springframework.cache.Cache 和 ...
- Poco XMLconfiguration 解析xml配置文件
环境: Centos7 GCC: 7.3.0 准备需要读取的xml文件: <config> <prop1>1.23</prop1> <prop2>2.3 ...
- 阿里云 RDS 数据库又发 CPU 近 100% 的“芯脏病”
最近云界发生了2件事,一件是大事,一件是小事,大事是阿里云与微软合作推出了开放应用模型 Open Application Model(OAM),小事是由于微软 SQL Server 在阿里云上水土不服 ...
- AOP框架Dora.Interception 3.0 [2]: 实现原理
和所有的AOP框架一样,我们必须将正常的方法调用进行拦截,才能将应用到当前方法上的所有拦截器纳入当前调用链.Dora.Interception采用IL Eimit的方式实现对方法调用的拦截,接下来我们 ...
- 2-SAT问题学习笔记+例题[洛谷P4792]
一个不错的2-SAT文章:传送门 问题初入 什么是2-SAT SAT是适定性(Satisfiability)问题的简称 .一般形式为k-适定性问题,简称 k-SAT. 首先,把「2」和「SAT」拆开. ...
- PowerShell渗透--Empire
0x00 简介 Empire是一款针对Windows平台的,使用PowerShell脚本作为攻击载荷的渗透攻击框架代码具有从stager生成,提权到渗透维持的一系列功能,无需powershell.ex ...
- MyCat教程二:mysql主从复制实现
单个mysql数据库在处理业务的时候肯定是有限的,这时我们扩展数据库的第一种方式就是对数据库做读写分离(主从复制),本文我们就先来介绍下怎么来实现mysql的主从复制操作. 1. 读写分离 原 ...
- Head First设计模式——装饰者模式
前言:对于设计模式我们有时候在想是否有必要,因为实际开发中我们没有那么多闲工夫去套用这么多设计模式,也没有必要为了模式而模式. 通常这些模式会引入新的抽象层,增加代码的复杂度,但是当我们掌握了这些设计 ...
- API设计中防重放攻击
HTTPS数据加密是否可以防止重放攻击? 否,加密可以有效防止明文数据被监听,但是却防止不了重放攻击. 防重放机制 我们在设计接口的时候,最怕一个接口被用户截取用于重放攻击.重放攻击是什么呢?就是把你 ...
- 使用JRebel插件实现SpringBoot应用代码热加载
前言 在实际的开发过程中,我们经常修改代码之后,手动的重启项目,查看修改效果.那么有没有一种方式能够快速的.自动的帮我们将修改代码自动更新,避免手动重启,从而提高开发效率呢?是有的,在我之前的文章里面 ...