CF 452E. Three strings(后缀数组+并查集)
解题思路
感觉这种题都是套路之类的??首先把三个串并成一个,中间插入一些奇怪的字符,然后跑遍\(SA\)。考虑按照\(height\)分组计算,就是每个\(height\)只在最高位计算一次,然后求个后缀和,这个可以并查集来实现。具体就是记一个\(sum[i][3]\)表示第\(i\)个联通块中\(0,1,2\)的个数,\(0,1,2\)就是出现在三个串的哪一个,然后合并时需要容斥一下。
代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
const int N=300005;
const int MOD=1e9+7;
char s[N];
int fa[N],sum[N][3],n,m,Len,id[N],height[N],ans[N],type[N];
struct SA{
int X[N<<1],Y[N<<1],rk[N],sa[N],c[N],num;
void get_SA(){
for(int i=1;i<=n;++i) X[i]=s[i],c[X[i]]++;
for(int i=2;i<=m;++i) c[i]+=c[i-1];
for(int i=n;i;--i) sa[c[X[i]]--]=i;
for(int k=1;k<=n;k<<=1){ num=0;
for(int i=n-k+1;i<=n;++i) Y[++num]=i;
for(int i=1;i<=n;++i) if(sa[i]>k) Y[++num]=sa[i]-k;
memset(c,0,sizeof(c));
for(int i=1;i<=n;++i) c[X[i]]++;
for(int i=2;i<=m;++i) c[i]+=c[i-1];
for(int i=n;i;--i) sa[c[X[Y[i]]]--]=Y[i],Y[i]=0;
swap(X,Y); X[sa[1]]=1; num=1;
for(int i=2;i<=n;++i)
X[sa[i]]=(Y[sa[i]]==Y[sa[i-1]] && Y[sa[i]+k]==Y[sa[i-1]+k])?num:++num;
m=num; if(m==n) break;
}
}
void get_height(){
for(int i=1;i<=n;++i) rk[sa[i]]=i;
int j,k=0;
for(int i=1;i<=n;++i){
if(rk[i]==1) continue;
if(k) k--; j=sa[rk[i]-1];
while(j+k<=n && i+k<=n && s[i+k]==s[j+k]) k++;
height[rk[i]]=k;
}
}
}t;
inline bool cmp(int x,int y){
return height[x]>height[y];
}
int get(int x){
if(x==fa[x]) return x;
return fa[x]=get(fa[x]);
}
int main(){
char ss[N]; int len;
scanf("%s",ss+1); len=strlen(ss+1); Len=len;
for(int i=1;i<=len;++i) s[++n]=ss[i],type[n]=1;
s[++n]='#'; type[n]=-1;
scanf("%s",ss+1); len=strlen(ss+1); Len=min(Len,len);
for(int i=1;i<=len;++i) s[++n]=ss[i],type[n]=2;
s[++n]='#'+1; type[n]=-1;
scanf("%s",ss+1); len=strlen(ss+1); Len=min(Len,len);
for(int i=1;i<=len;++i) s[++n]=ss[i],type[n]=3;
m='z'; t.get_SA(); t.get_height();
for(int i=1;i<=n;i++) id[i]=i,fa[i]=i;
for(int i=1;i<=n;i++){
if(type[i]==1) sum[i][0]=1;
if(type[i]==2) sum[i][1]=1;
if(type[i]==3) sum[i][2]=1;
}
sort(id+1,id+1+n,cmp); int now=1,tmp=0,u,v;
for(int i=Len;i;i--){
while(now<=n && height[id[now]]>=i) {
u=get(t.sa[id[now]-1]),v=get(t.sa[id[now]]);
tmp-=(1ll*sum[u][0]*sum[u][1]%MOD*sum[u][2]%MOD);
tmp=(tmp+MOD)%MOD;
tmp-=(1ll*sum[v][0]*sum[v][1]%MOD*sum[v][2]%MOD);
tmp=(tmp+MOD)%MOD;
sum[u][0]=(sum[u][0]+sum[v][0])%MOD;
sum[u][1]=(sum[u][1]+sum[v][1])%MOD;
sum[u][2]=(sum[u][2]+sum[v][2])%MOD;
tmp=tmp+(1ll*sum[u][0]*sum[u][1]%MOD*sum[u][2]%MOD)%MOD;
tmp%=MOD; now++; fa[v]=u;
}
ans[i]=tmp;
}
for(int i=1;i<=Len;i++) printf("%d ",ans[i]);
return 0;
}
CF 452E. Three strings(后缀数组+并查集)的更多相关文章
- BZOJ 4566 JZYZOJ 1547 [haoi2016T5]找相同子串 后缀数组 并查集
http://172.20.6.3/Problem_Show.asp?id=1547 http://www.lydsy.com/JudgeOnline/problem.php?id=4566 单纯后缀 ...
- NOI 2015 品酒大会 (后缀数组+并查集)
题目大意:略 40分暴力还是很好写的,差分再跑个后缀和 和 后缀最大值就行了 一种正解是后缀数组+并查集 但据说还有后缀数组+单调栈的高端操作蒟蒻的我当然不会 后缀数组求出height,然后从大到小排 ...
- [UOJ#131][BZOJ4199][NOI2015]品酒大会 后缀数组 + 并查集
[UOJ#131][BZOJ4199][NOI2015]品酒大会 试题描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个 ...
- Uva 12361 File Retrieval 后缀数组+并查集
题意:有F个单词,1 <= F <=60 , 长度<=10^4, 每次可以输入一个字符串,所有包含该字串的单词会形成一个集合. 问最多能形成多少个不同的集合.集合不能为空. 分析:用 ...
- BZOJ 4199: [Noi2015]品酒大会( 后缀数组 + 并查集 )
求出后缀数组后, 对height排序, 从大到小来处理(r相似必定是0~r-1相似), 并查集维护. 复杂度O(NlogN + Nalpha(N)) ------------------------- ...
- 【学术篇】NOI2015 品酒大会 后缀数组+并查集
省选前大致是刷不了几道题了... 所以就找一些裸一点的题目练练板子算了= = 然而这题一点都不裸, 也并不怎么好写... 于是就浪费了将近一下午的时间... 然而还不是因为后缀数组板子不熟= = 首先 ...
- POJ 3415 Common Substrings 后缀数组+并查集
后缀数组,看到网上很多题解都是单调栈,这里提供一个不是单调栈的做法, 首先将两个串 连接起来求height 求完之后按height值从大往小合并. height值代表的是 sa[i]和sa[i ...
- 4199. [NOI2015]品酒大会【后缀数组+并查集】
Description 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品 酒家”和“首席猎手”两个奖项,吸引了众多品酒师参加.在大会的晚餐上,调酒师 ...
- 【BZOJ4199】[Noi2015]品酒大会 后缀数组+并查集
[BZOJ4199][Noi2015]品酒大会 题面:http://www.lydsy.com/JudgeOnline/wttl/thread.php?tid=2144 题解:听说能用SAM?SA默默 ...
随机推荐
- memcached php扩展(二)
memcached php扩展(二) 安装环境链接:http://pan.baidu.com/s/1i4IbJox Memecached 服务器安装(一) memcached php扩展(二) red ...
- Oracle客户端和服务端的区别
转载:https://blog.csdn.net/qq_22558507/article/details/75220224 随着Oracle技术领域应用越来越广泛,现在就关于oracle客户端配置极其 ...
- 远控CVE整理
Windows: CVE-2017-8464(通过快捷方式,可U盘/共享等途径传播)
- mooc-IDEA 调试代码--012
mooc-IDEA 调试代码 添加断点快捷键:ctrl+F8 单步运行:F9 <=>resum(从一个断点跳转到下一个断点) 一行一行运行:F8 查看所有断点: 禁止所有断点: 条件断点 ...
- 《JAVA设计模式》之单例模式(Singleton)
在阎宏博士的<JAVA与模式>一书中开头是这样描述单例模式的: 作为对象的创建模式,单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类. 单例模式的 ...
- [eclipse中使用Git插件] 008 - git操作pull、merge、stash、commit
写在前面: 看标题其实还有001-007之类,本来准备写详细的类似教程一样的东东,但是懒了且时间有限(以后或许会补吧),所以跳到008,录下主要的操作. 所以本随笔的重点就是[pull+merge+s ...
- [Codeforces 865C]Gotta Go Fast(期望dp+二分答案)
[Codeforces 865C]Gotta Go Fast(期望dp+二分答案) 题面 一个游戏一共有n个关卡,对于第i关,用a[i]时间通过的概率为p[i],用b[i]通过的时间为1-p[i],每 ...
- python常用模块---collections、time、random、os、sys、序列号模块
collections模块 在内置数据类型(dict.list.set.tuple)的基础上,collections模块还提供了几个额外的数据类型:Counter.deque.defaultdict. ...
- 【JMeter5.0】Mac安装JDK和JMeter5
之前讲了Windows下安装JDK和JMeter4.0的方法,其实不论操作系统是Windows.Mac OS.Linux等,JMeter所需要的基础环境配置都是类似的,本文介绍JMeter for M ...
- 【JMeter4.0】一、JAVA环境-JDK1.10安装与配置
环境变量的作用: 它是操作系统用来指定运行环境的一些参数.比如临时文件夹位置和系统文件夹位置等.当你运行某些程序时,除了在当前文件夹中寻找外,还会到这些环境变量中去查找,比如“Path”就是一个变量, ...