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默默 ...
随机推荐
- C# DataTable删除行Delete与Remove的问题
DataTable删除行使用Delete后,只是该行被标记为deleted,但是还存在,用Rows.Count来获取行数时,还是删除之前的行数,需要使用datatable.AcceptChanges( ...
- Python笔记(十六)_else语句、with语句
else的多种用法 1.try except + else:检测到代码无异常,才执行else 例如: def func(num): count=num//2 while count>1: if ...
- 10.jmeter jsr223 javascript 深度比对json object
function sortJSON(data, key, way) { //log.info(" " + key + " ------------------- &quo ...
- IDF-CTF-简单的js加密 writeup
题目链接: http://ctf.idf.cn/index.php?g=game&m=article&a=index&id=43 知识点:js语法 这里这里→ http://c ...
- 《JAVA设计模式》之责任链模式(Chain of Responsibility)
在阎宏博士的<JAVA与模式>一书中开头是这样描述责任链(Chain of Responsibility)模式的: 责任链模式是一种对象的行为模式.在责任链模式里,很多对象由每一个对象对其 ...
- zookeeper+dubbo+demo
zookeeper下载地址 https://archive.apache.org/dist/zookeeper/ zookeeper安装和使用 windows环境 https://blog.csdn. ...
- 自己做的html5手机站点
这个站点採用html5+css3+jquerymobile 仅仅开发了前台的功能,前台的界面比較美观,后端不方便放上,各位有空能够自己开发这个站点的界面有參考各个菜谱站点的页面的样式,可是又不一样,是 ...
- Java-技术专区-设计模式-reactor模式
模型: 反应器模式做法是:汽车是乘客访问的主体(Reactor),乘客上车后,到售票员(acceptor)处登记,之后乘客便可以休息睡觉去了,当到达乘客所要到达的目的地后,售票员将其唤 ...
- LeetCode 852. Peak Index in a Mountain Array(C++)
Let's call an array A a mountain if the following properties hold: A.length >= 3 There exists som ...
- shell脚本从入门到精通(中级)之提高篇
shell 脚本入门到精通(中级) 一.shell 脚本的执行 二.输出格式化 三.数据类型 四.重定向 五.变量 一.shell 脚本的执行 1. 脚本执行的4种方法 $ ls /tmp/test. ...