luogu1117 优秀的拆分 (后缀数组)
考虑分别计算每个位置作为AA的末尾或者BB的开头的个数 最后乘一乘就是答案
据说是套路的计算AA的方法:
首先枚举A的长度L,然后每L个字符当做一个关键点,这样的话,一个AA包含且只包含相邻两个关键点(记为a,b),而且满足lcp(a,b)+lcs(a,b)-1>=L 手画一下就能看出来
于是SA搞lcp 倒过来再SA搞lcs 最后差分一下统计答案即可
#include<bits/stdc++.h>
#define pa pair<int,int>
#define CLR(a,x) memset(a,x,sizeof(a))
#define MP make_pair
using namespace std;
typedef long long ll;
const int maxn=3e4+; inline char gc(){
return getchar();
static const int maxs=<<;static char buf[maxs],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,,maxs,stdin),p1==p2)?EOF:*p1++;
}
inline ll rd(){
ll x=;char c=gc();bool neg=;
while(c<''||c>''){if(c=='-') neg=;c=gc();}
while(c>=''&&c<='') x=(x<<)+(x<<)+c-'',c=gc();
return neg?(~x+):x;
} char str[maxn];
int lg2[maxn],N; struct SA{
int cnt[maxn*],tmp[maxn*],rnk[maxn*],sa[maxn*],hei[maxn];
int st[maxn][],n; inline void build(char *s){
int i,j=,k,m=;
CLR(cnt,);CLR(rnk,);
for(i=;i<=n;i++) cnt[s[i]-'a']=;
for(i=;i<=m;i++) cnt[i]+=cnt[i-];
for(i=;i<=n;i++) rnk[i]=cnt[s[i]-'a'];
for(k=;j!=n;k<<=){
CLR(cnt,);
for(i=;i<=n;i++) cnt[rnk[i+k]]++;
for(i=;i<=m;i++) cnt[i]+=cnt[i-];
for(i=n;i;i--) tmp[cnt[rnk[i+k]]--]=i;
CLR(cnt,);
for(i=;i<=n;i++) cnt[rnk[i]]++;
for(i=;i<=m;i++) cnt[i]+=cnt[i-];
for(i=n;i;i--) sa[cnt[rnk[tmp[i]]]--]=tmp[i];
memcpy(tmp,rnk,sizeof(rnk));
rnk[sa[]]=j=;
for(i=;i<=n;i++){
if(tmp[sa[i]]!=tmp[sa[i-]]||tmp[sa[i]+k]!=tmp[sa[i-]+k]) j++;
rnk[sa[i]]=j;
}m=j;
} hei[]=;
for(i=,j=;i<=n;i++){
if(rnk[i]==) continue;
if(j) j--;
int x=sa[rnk[i]-];
while(i+j<=n&&x+j<=n&&s[i+j]==s[x+j]) j++;
hei[rnk[i]]=j;
}
// for(i=1;i<=n;i++) printf("hei:%d %d ; rnk:%d ; sa:%d \n",i,hei[i],rnk[i],sa[i]);
for(i=n;i;i--){
st[i][]=hei[i];
for(j=;i+(<<j)-<=n;j++){
st[i][j]=min(st[i][j-],st[i+(<<(j-))][j-]);
}
}
} inline int query(int x,int y){ //x,y:pos
if(x==y) return n-y+;
int l=min(rnk[x],rnk[y])+,r=max(rnk[x],rnk[y]);
// printf("~%d %d\n",l,r);
int k=lg2[r-l+];
return min(st[l][k],st[r-(<<k)+][k]);
}
}fw,bw; int end[maxn],beg[maxn]; int main(){
//freopen("","r",stdin);
int i,j,k;
for(i=;i<=;i++) lg2[i]=lg2[i>>]+;
for(int T=rd();T;T--){
scanf("%s",str+);N=strlen(str+);
fw.n=bw.n=N;
fw.build(str);
reverse(str+,str+N+);
bw.build(str);
CLR(end,);CLR(beg,);
for(int l=;l<=N;l++){
i=l+;
for(;i<=N;i+=l){
int lcp=min(l,fw.query(i-l,i)),lcs=min(l,bw.query(N-i+,N-(i-l)+));
// printf("!%d %d %d %d\n",i,i-l,lcp,lcs);
if(lcp+lcs->=l){
end[i+l-lcs]++,end[i+lcp]--;
// printf("add A:%d %d\n",i+l-lcs,i+lcp-1);
beg[i-l-lcs+]++,beg[i-*l+lcp+]--;
// printf("add B:%d %d\n",i-l-lcs+1,i-2*l+lcp);
}
}
}
for(i=;i<=N;i++) end[i]+=end[i-],beg[i]+=beg[i-];
ll ans=;
for(i=;i<N;i++){
ans+=end[i]*beg[i+];
}
printf("%lld\n",ans);
}
return ;
}
luogu1117 优秀的拆分 (后缀数组)的更多相关文章
- NOI 2016 优秀的拆分 (后缀数组+差分)
题目大意:给你一个字符串,求所有子串的所有优秀拆分总和,优秀的拆分被定义为一个字符串可以被拆分成4个子串,形如$AABB$,其中$AA$相同,$BB$相同,$AB$也可以相同 作为一道国赛题,95分竟 ...
- [NOI2016]优秀的拆分 后缀数组
题面:洛谷 题解: 因为对于原串的每个长度不一定等于len的拆分而言,如果合法,它将只会被对应的子串统计贡献. 所以子串这个限制相当于是没有的. 所以我们只需要对于每个位置i求出f[i]表示以i为开头 ...
- UOJ#219. 【NOI2016】优秀的拆分 [后缀数组 ST表]
#219. [NOI2016]优秀的拆分 题意:求有多少AABB样子的子串,拆分不同的同一个子串算多个 一开始一直想直接求,并不方便 然后看了一眼Claris的题解的第一行就有思路了 如果分开,求\( ...
- BZOJ.4650.[NOI2016]优秀的拆分(后缀数组 思路)
BZOJ 洛谷 令\(st[i]\)表示以\(i\)为开头有多少个\(AA\)这样的子串,\(ed[i]\)表示以\(i\)结尾有多少个\(AA\)这样的子串.那么\(Ans=\sum_{i=1}^{ ...
- UOJ #219 BZOJ 4650 luogu P1117 [NOI2016]优秀的拆分 (后缀数组、ST表)
连NOI Day1T1都不会做...看了题解都写不出来还要抄Claris的代码.. 题目链接: (luogu)https://www.luogu.org/problemnew/show/P1117 ( ...
- BZOJ 4650 [Noi2016]优秀的拆分 ——后缀数组
我们只需要统计在某一个点开始的形如$AA$字符串个数,和结束的个数相乘求和. 首先枚举循环节的长度L.即$\mid (A) \mid=L$ 然后肯定会经过s[i]和[i+L]至少两个点. 然后我们可以 ...
- [NOI2016]优秀的拆分(SA数组)
[NOI2016]优秀的拆分 题目描述 如果一个字符串可以被拆分为 \(AABB\) 的形式,其中 A和 B是任意非空字符串,则我们称该字符串的这种拆分是优秀的. 例如,对于字符串 \(aabaaba ...
- 【BZOJ4650】【NOI2016】优秀的拆分(后缀数组)
[BZOJ4650][NOI2016]优秀的拆分(后缀数组) 题面 BZOJ Uoj 题解 如果我们知道以某个位置为开始/结尾的\(AA\)串的个数 那就直接做一下乘法就好 这个怎么求? 枚举一个位置 ...
- 【BZOJ4650】[NOI2016] 优秀的拆分(后缀数组)
点此看题面 大致题意: 定义将一个字符串拆成\(AABB\)的形式为优秀拆分,求一个字符串所有子串的优秀拆分个数. 后缀数组 这题可是一道后缀数组黑题啊. 其实看完题解这题还是挺简单的. 大致思路 显 ...
随机推荐
- linux下磁盘存储空间不足
把自己平时遇到的问题分享给大家 Question:linux系统Ubuntu下面有一个Trash,当我们删除文件后,清空Trash,过一段时间发现磁盘空间不足 Answer:其实我们并没有真正的删除文 ...
- 微信小程序转发微信小程序转发
微信小程序转发涉及以下4个方法: 1.Page.onShareAppMessage({}) 设置右上角“转发”配置,及转发后回调函数返回 shareTicket 票据 2.wx.showSahreMe ...
- sql取指定时间段内的所有月份
declare @begin datetime,@end datetime set @begin='2017-01-01' set @end='2019-03-04' declare @months ...
- c/c++ 头文件的血案
头文件的血案 不小心在一个头文件里,加了函数的定义,结果导致编译时,提示这个函数被重复定义:( Quote.h #ifndef __QUOTE_H__ #define __QUOTE_H__ #inc ...
- Java - String 的字面量、常量池、构造函数和intern()函数
一.内存中的 String 对象 Java 的堆和栈 对于基本数据类型变量和对象的引用,也就是局部变量表属于栈内存: 而通过 new 关键字和 constructor 创建的对象存放在堆内存: 直接的 ...
- powerdesigner生成mysql带注释的ER图
1.安装PowerDesigner的 参考 https://blog.csdn.net/sinat_34104446/article/details/79885141 2配置逆向工程 2.1新建模型p ...
- 【PAT】A1034Head of a Gang
昨天准备学完图相关的知识,但是学起来挺懵的,理解起来不难,但自己一回想,又什么都想不起来. 翻来覆去看图的遍历,还是觉得有点没到位. 所以做题来检测一下,果然学和自己做是两码事. 先看的书,又看的柳婼 ...
- const关键字到底该怎么用
原文地址:https://www.yanbinghu.com/2019/01/28/7442.html 前言 我们都知道使用const关键字限定一个变量为只读,但它是真正意义上的只读吗?实际中又该如何 ...
- day2 and day3 总结-python基础-坚持就是胜利
今日份快捷键学习,pycharm中按CTRL+ALT+L 自动规范化操作,不过和QQ的快捷键会有冲突,建议更改QQ的 知识点: 1.编码 2.while循环 3.运算符 4.数字int 5.布尔值 6 ...
- android 实现点击edittext的“小眼睛”切换明密文
android 实现点击edittext的“小眼睛”切换明密文 版权声明:本文为博主原创文章,未经博主允许不得转载. 很多时候,我们为了用户的隐私安全,需要在密码输入的时候,显示密文.为了更 ...