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默默 ...
随机推荐
- 传统神经网络ANN训练算法总结 参考 。 以后研究
http://blog.163.com/yuyang_tech/blog/static/21605008320146451352506/ 传统神经网络ANN训练算法总结 2014-07-04 17:1 ...
- JSP表单提交 与 接受显示
Demo01.jsp 提交表单输入的信息至 Demo02.jsp方法一 1 <%@ page language="java" contentType="text/h ...
- org.w3c.dom。 XML解析 练习
HTML文档 1 import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; ...
- 【ABAP系列】SAP ABAP基础-程序优化及响应速度之LOOP
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[ABAP系列]SAP ABAP基础-程序优化及 ...
- Codeforces Round #410 (Div. 2)B. Mike and strings(暴力)
传送门 Description Mike has n strings s1, s2, ..., sn each consisting of lowercase English letters. In ...
- netcore之mysql中文乱码问题解决记录
尝试了netcore代码里面设置基本无效了 https://dev.mysql.com/doc/connector-net/en/connector-net-entityframework-core- ...
- jmeter 非GUI执行测试,没有响应数据保存到jtl文件办法
估计是jmeter为了减轻客户机负担,就没又默认把这些信息保存,如果想要保存,也可以,需要做出如下配置: 修改bin目录下的user.properties文件,追加配置: jmeter.save.sa ...
- Spring Boot静态资源
1.4 SpringBoot静态资源 1.4.1 默认静态资源映射 Spring Boot 对静态资源映射提供了默认配置 Spring Boot 默认将 /** 所有访问映射到以下目录: classp ...
- 二叉树BinTree类定义
#include<iostream> using namespace std; template<class T> struct BinTreeNode{//二叉树结点类 T ...
- BZOJ 1937 (luogu 4412) (KM+LCA)
题面 传送门 分析 根据贪心的思想我们得到几条性质: 1.生成树上的边权减小,非树边的边权增加 2.每条边最多被修改一次 设改变量的绝对值为d 对于一条非树边\(j:(u,v)\),树上u->v ...