[bzoj 4650][NOI 2016]优秀的拆分
传送门
Description
如果一个字符串可以被拆分为\(AABB\) 的形式,其中$ A$和 \(B\)是任意非空字符串,则我们称该字符串的这种拆分是优秀的。
例如,对于字符串\(aabaabaa\),如果令\(A=aab\),\(B=a\),我们就找到了这个字符串拆分成 \(AABB\)的一种方式。
一个字符串可能没有优秀的拆分,也可能存在不止一种优秀的拆分。比如我们令\(A=a\),\(B=baa\),也可以用 \(AABB\)表示出上述字符串;但是,字符串 \(abaabaa\) 就没有优秀的拆分。
现在给出一个长度为 \(n\)的字符串\(S\),我们需要求出,在它所有子串的所有拆分方式中,优秀拆分的总个数。这里的子串是指字符串中连续的一段。
以下事项需要注意:
- 出现在不同位置的相同子串,我们认为是不同的子串,它们的优秀拆分均会被记入答案。
- 在一个拆分中,允许出现\(A=B\)。例如 \(cccc\) 存在拆分\(A=B=c\) 。
- 字符串本身也是它的一个子串。
Solution
\(st[i]\)表示以\(i\)开始的有多少个形式如\(aa\)的子串
\(en[i]\)表示以\(j\)结束的有多少个形式如\(aa\)的字串
那么答案就是\(\sum_{i=1}^{n-1} en[i]*st[i+1]\)
怎么求这个东西呢?只考虑\(st\),因为把原字符串倒个序就能求\(en\)了
枚举\(aa\)中\(a\)的长度,然后每次我们只考虑\([ka+1,(k+1)a]\)这个区间上有多少个可以作为起始点的,它一定包含\((k+1)a\)这个位置,而后面与它相同的字串一定包括\((k+2)a\)这个位置,所以直接查询\(LCP((k+1)a,(k+2)a)\),以及\(LCS((k+1)a,(k+2)a)\),就可以啦。
根据调和级数和\(SA\),总复杂度是\(O(n\log n)\)
Code
#include<bits/stdc++.h>
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
inline int read()
{
int x=0,f=1;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;
}
#define MN 30005
#define M(a) memset(a,0,sizeof a);
char s[MN];
int lg[MN],en[MN],st[MN],n;
void init(){n=strlen(s+1);M(st);M(en);}
class SA
{
private:
int height[MN][19],p,q,sa[2][MN],rk[2][MN],num[MN];
public:
SA(){M(height);M(sa);M(rk);M(num);}
inline void init(){M(height);M(sa);M(rk);M(num);}
inline void build_sa()
{
register int i,j,k,mx;
for(i=1;i<=n;++i) num[s[i]-'a'+1]++;
for(i=1;i<=26;++i) num[i]+=num[i-1];
for(i=1;i<=n;++i) sa[1][num[s[i]-'a'+1]--]=i;
for(i=1;i<=n;++i) rk[1][sa[1][i]]=rk[1][sa[1][i-1]]+(s[sa[1][i-1]]!=s[sa[1][i]]);
mx=rk[1][sa[1][n]];
for(p=1,q=0,k=1;k<=n;k<<=1,p^=1,q^=1)
{
if(mx==n) break;
for(i=1;i<=n;++i) num[rk[p][sa[p][i]]]=i;
for(i=n;i;--i) if(sa[p][i]>k) sa[q][num[rk[p][sa[p][i]-k]]--]=sa[p][i]-k;
for(i=n-k+1;i<=n;++i) sa[q][num[rk[p][i]]--]=i;
for(i=1;i<=n;++i)
rk[q][sa[q][i]]=rk[q][sa[q][i-1]]+(rk[p][sa[q][i]]!=rk[p][sa[q][i-1]]||rk[p][sa[q][i]+k]!=rk[p][sa[q][i-1]+k]);
mx=rk[q][sa[q][n]];
}
for(i=k=1;i<=n;++i)
{
if(rk[p][i]==1) continue;if(k) k--;
for(j=sa[p][rk[p][i]-1];j+k<=n&&i+k<=n&&s[i+k]==s[j+k];++k);
height[rk[p][i]][0]=k;
}
for(i=1;i<=18;++i)for(j=n;j>=1&&j>(1<<i);--j)
height[j][i]=min(height[j][i-1],height[j-(1<<i-1)][i-1]);
}
inline int LCP(int x,int y)
{
x=rk[p][x];y=rk[p][y];
if(x>y) std::swap(x,y);
return min(height[y][lg[y-x]],height[x+(1<<lg[y-x])][lg[y-x]]);
}
}A,revA;
int main()
{
register int T,i,j;
for(i=2;i<MN;++i) lg[i]=lg[i>>1]+1;
T=read();printf("%d\n",T);
while(T--)
{
memset(s,0,sizeof s);
scanf("%s",s+1);
init();
A.init();A.build_sa();
for(i=1;i+i<=n;++i) std::swap(s[i],s[n-i+1]);
revA.init();revA.build_sa();
register int len,x,y;
for(len=1;len<=n;++len)for(i=len,j=len<<1;j<=n;i+=len,j+=len)
{
x=min(A.LCP(i,j),len);y=min(revA.LCP(n-i+1,n-j+1),len);
if(x+y-1<len) continue;
st[i-y+1]++;st[i+x-len+1]--;
en[j+len-y]++;en[j+x]--;
}
for(i=1;i<=n;++i) st[i]+=st[i-1],en[i]+=en[i-1];
register ll ans=0;
for(i=1;i<n;++i) ans+=1ll*en[i]*st[i+1];
printf("%lld\n",ans);
}
return 0;
}
Blog来自PaperCloud,未经允许,请勿转载,TKS!
[bzoj 4650][NOI 2016]优秀的拆分的更多相关文章
- [LOJ 2083][UOJ 219][BZOJ 4650][NOI 2016]优秀的拆分
[LOJ 2083][UOJ 219][BZOJ 4650][NOI 2016]优秀的拆分 题意 给定一个字符串 \(S\), 求有多少种将 \(S\) 的子串拆分为形如 AABB 的拆分方案 \(| ...
- NOI 2016 优秀的拆分 (后缀数组+差分)
题目大意:给你一个字符串,求所有子串的所有优秀拆分总和,优秀的拆分被定义为一个字符串可以被拆分成4个子串,形如$AABB$,其中$AA$相同,$BB$相同,$AB$也可以相同 作为一道国赛题,95分竟 ...
- 字符串(后缀自动机):NOI 2016 优秀的拆分
[问题描述] 如果一个字符串可以被拆分为 AABB 的形式,其中 A 和 B 是任意非空字符串, 则我们称该字符串的这种拆分是优秀的. 例如,对于字符串 aabaabaa,如果令 A = aab, B ...
- [NOI 2016]优秀的拆分
Description 题库链接 给你一个长度为 \(n\) 的只含小写字母的字符串 \(S\) ,计算其子串有多少优秀的拆分. 如果一个字符串能被表示成 \(AABB\) 的形式,其中 \(A,B\ ...
- 解题:NOI 2016 优秀的拆分
题面 其实题目不算很难,但是我调试的时候被玄学了,for循环里不写空格会RE,写了才能过.神**调了一个多小时是这么个不知道是什么的玩意(真事,可以问i207M=.=),心态爆炸 发现我们只要找AA或 ...
- [BZOJ]4650 优秀的拆分(Noi2016)
比较有意思的一道后缀数组题.(小C最近是和后缀数组淦上了?) 放在NOI的考场上.O(n^3)暴力80分,O(n^2)暴力95分…… 即使想把它作为一道签到题也不要这么随便啊摔(╯‵□′)╯︵┻━┻ ...
- [BZOJ]4650: [Noi2016]优秀的拆分
Time Limit: 30 Sec Memory Limit: 512 MB Description 如果一个字符串可以被拆分为 AABBAABB 的形式,其中 AA 和 BB 是任意非空字符串, ...
- 【NOI 2016】优秀的拆分
Problem Description 如果一个字符串可以被拆分为 \(AABB\) 的形式,其中 \(A\) 和 \(B\) 是任意非空字符串,则我们称该字符串的这种拆分是优秀的. 例如,对于字符串 ...
- 【刷题】BZOJ 4650 [Noi2016]优秀的拆分
Description 如果一个字符串可以被拆分为 AABBAABB 的形式,其中 AA 和 BB 是任意非空字符串,则我们称该字符串的这种拆分是优秀的.例如,对于字符串 aabaabaa,如果令 A ...
随机推荐
- 第2章 NIO入门
2.1 传统的BIO编程 以服务器为例,在传统BIO模型下的服务器,每当一个新的请求到来的时候回分配一个线程去处理该请求,并且该线程在执行IO操作的时候会一直阻塞,知道IO操作完成或抛出异常才会返回. ...
- Python之算法评估-4
一.评估算法的方式分两种,一种是分类算法的评估,一种是回归算法的评估.为什么要分两种呢,因为分类算法中可以通过准确率.精准率.召回率.混淆矩阵.AUC来评估算法的准确度.但是在预测值的时候是没有办法去 ...
- Java定时任务工具详解之Timer篇
Java定时任务调度工具详解 什么是定时任务调度? ◆ 基于给定的时间点,给定的时间间隔或者给定的执行次数自动执行的任务. 在Java中的定时调度工具? ◆ Timer ◆Quartz T ...
- ef core数据迁移的一点小感悟
ef core在针对mysql数据迁移的时候,有些时候没法迁移...有两种情况没法迁移,一种是因为efcore的bug问题导致没法迁移,这个在github上有个问题集,另外一种是对数据表进行较大幅度的 ...
- 同步IO,异步IO,阻塞,非阻塞的定义与区别
异步I/O 是指用户程序发起IO请求后,不等待数据,同时操作系统内核负责I/O操作把数据从内核拷贝到用户程序的缓冲区后通知应用程序.数据拷贝是由操作系统内核完成,用户程序从一开始就没有等待数据,发起请 ...
- springboot笔记04——读取配置文件+使用slf4j日志
前言 springboot常用的配置文件有yml和properties两种,当然有必要的时候也可以用xml.我个人更加喜欢用yml,所以我在这里使用yml作为例子.yml或properties配置文件 ...
- echarts3关系图:力引导布局, 固定某些节点
在数组里设置 fixed: true,<a href='http://echarts.baidu.com/option.html#series-graph.data.fixed'>官方文档 ...
- cmd修改路径
1. 切换盘符: 输入X:(例如C:)+回车 2. 查看当前目录下的文件:输入dir+回车 3. 选择想要的文件,输入:cd+文件名
- 剑指offer-链表相关
剑指offer面试常考手撸算法题-链表篇 1. 从头到尾打印链表 class Solution { public: // 可以先压栈,再出栈到vector // 时间/空间:O(n) vector&l ...
- 利用EasyUI 数据网格(DataGrid)的loader属性实现后端分页
该属性在easyui官方文档中并没有详细阐述,通过简单的资料查询和摸索,实现了easyUI数据网格的后端分页功能. 在官网文档中这样阐述loader属性: 定义如何从远程服务器加载数据.返回false ...