题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3277

   https://www.lydsy.com/JudgeOnline/problem.php?id=3473

学习的博客:https://www.cnblogs.com/HocRiser/p/9580478.html

广义后缀自动机有两种写法,这里写的是 trie 树的那种。

大意就是每个串从自动机的根开始走,

1.如果存在 q = go[p][w] ,且 l [q] == l [p]+1 ,  那么直接把 q 看作这次插入了的点,下次令 p = q ,续着往后插入;

2.如果存在 q = go[p][w] ,且 l [q] != l [p]+1 , 那么分出一个 nq 来,把 nq 看作这次插入了的点,下次令 p = nq ,续着往后插入;

3.如果不存在 q = go[p][w] ,就新建一个 np ;然后就是一个串的后缀自动机插入了;下次令 p = np 。

要计算后缀自动机上的每个点代表的那些子串在所有串里出现了多少次,记为 ct[ ] ;

  在做一个串的插入的时候,考虑让一些位置的 ct[ ] ++ ;每插入一个字符,就跳 fa ,给所有 fa 的 ct[ ] ++ ;但如果已经被当前字符串的之前字符弄得 ct[ ] 加一过了,就不用再加了;

  不要边建自动机边做 ct[ ] ++ ;因为那时的 parent 树还不是所有串的。

然后拓扑排序一下,从根到叶子做 dp ,自己节点可以贡献的合法子串个数 ans[ ] 就是 ans[ fa ] 再加上 “ 自己出现了 >= k 次?l [cr] - l [fa] : 0 ” 。

两道题用同样的代码即可。

#include<cstdio>
#include<cstring>
#include<string>//
#include<algorithm>
#define ll long long
using namespace std;
int Mx(int a,int b){return a>b?a:b;}
const int N=1e5+,M=N<<,K=;//<<1
int n,k,go[M][K],l[M],fa[M],tot=;
int tx[M],q[M],ans[M],ct[M],vis[M];
int cz(int p,int w)
{
int q=go[p][w],nq=++tot;l[nq]=l[p]+;
fa[nq]=fa[q];fa[q]=nq;
memcpy(go[nq],go[q],sizeof go[q]);
for(;p&&go[p][w]==q;p=fa[p])go[p][w]=nq;
return nq;
}
int ins(int p,int w)
{
if(go[p][w])
{
int q=go[p][w];
if(l[q]==l[p]+)return go[p][w];
else return cz(p,w);//////
}
else
{
int np=++tot;l[np]=l[p]+;
for(;p&&!go[p][w];p=fa[p])go[p][w]=np;
if(!p)fa[np]=;
else
{
int q=go[p][w];
if(l[q]==l[p]+)fa[np]=q;
else fa[np]=cz(p,w);
}
return np;
}
}
void Rsort(int mxn)
{
for(int i=;i<=tot;i++)tx[l[i]]++;
for(int i=;i<=mxn;i++)tx[i]+=tx[i-];
// for(int i=tot;i;i--)q[tx[l[i]]--]=i;
for(int i=;i<=tot;i++)q[tx[l[i]]--]=i;
}
string ch[N]; int len[N];
int main()
{
scanf("%d%d",&n,&k); char tp[N]; int mxn=;
for(int i=,d;i<=n;i++)
{
scanf("%s",tp);len[i]=strlen(tp);mxn=Mx(mxn,len[i]);
ch[i]=tp;
}
for(int i=;i<=n;i++)
for(int lm=len[i],pr=,j=;j<lm;j++)
pr=ins(pr,ch[i][j]-'a'+);
for(int i=;i<=n;i++)
for(int lm=len[i],cr=,j=;j<lm;j++)
{
cr=go[cr][ch[i][j]-'a'+];
for(int p=cr;p&&vis[p]!=i;p=fa[p])
ct[p]++,vis[p]=i;
}
Rsort(mxn);
for(int i=,d;i<=tot;i++)
{d=q[i];ans[d]=ans[fa[d]]+(ct[d]>=k?l[d]-l[fa[d]]:);}//
for(int i=;i<=n;i++)
{
ll prn=;
for(int lm=len[i],cr=,j=;j<lm;j++)
{
cr=go[cr][ch[i][j]-'a'+];
prn+=ans[cr];
}
printf("%lld ",prn);
}
puts("");return ;
}

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2780

和上一道题一样。

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<string>
using namespace std;
int const xn=2e5+;
const int N=1e4+,M=2e5+,N2=4e5+,K=;
int n,go[M][K],fa[M],l[M],ct[M],tot=;
string s[N];int len[N],vis[M];
int cz(int p,int w)
{
int q=go[p][w],nq=++tot; l[nq]=l[p]+;/////////l[p]+1 not l[q]+1!!!!!
fa[nq]=fa[q];fa[q]=nq;
memcpy(go[nq],go[q],sizeof go[q]);
for(;p&&go[p][w]==q;p=fa[p])go[p][w]=nq;
return nq;
}
int ins(int p,int w)
{
if(go[p][w])
{
int q=go[p][w];
if(l[q]==l[p]+)return q; else return cz(p,w);
}
else
{
int np=++tot;l[np]=l[p]+;
for(;p&&!go[p][w];p=fa[p])go[p][w]=np;
if(!p)fa[np]=;
else
{
int q=go[p][w];
if(l[q]==l[p]+)fa[np]=q; else fa[np]=cz(p,w);
}
return np;
}
}
char ch[N2]; int Q;
int main()
{
scanf("%d%d",&n,&Q);
for(int i=;i<=n;i++)
{
scanf("%s",ch);len[i]=strlen(ch);
s[i]=(string)ch;
for(int pr=,j=,lm=len[i];j<lm;j++)
pr=ins(pr,ch[j]-'a'+);
}
for(int i=;i<=n;i++)
for(int cr=,j=,lm=len[i];j<lm;j++)
{
cr=go[cr][s[i][j]-'a'+];
for(int p=cr;p>&&vis[p]!=i;p=fa[p])
ct[p]++,vis[p]=i;
}
while(Q--)
{
scanf("%s",ch);int lm=strlen(ch),cr=;
for(int j=;j<lm&&cr;j++)
cr=go[cr][ch[j]-'a'+];
printf("%d\n",ct[cr]);
}
return ;
}

bzoj 3277 串 && bzoj 3473 字符串 && bzoj 2780 [Spoj]8093 Sevenk Love Oimaster——广义后缀自动机的更多相关文章

  1. BZOJ 2780 [Spoj]8093 Sevenk Love Oimaster ——广义后缀自动机

    给定n个串m个询问,问每个串在n个串多少个串中出现了. 构建广义后缀自动机,(就是把所有字符串的后缀自动机合并起来)其实只需要add的时候注意一下就可以了. 然后对于每一个串,跑一边匹配,到达了now ...

  2. BZOJ 2780: [Spoj]8093 Sevenk Love Oimaster [广义后缀自动机]

    JZPGYZ - Sevenk Love Oimaster     Oimaster and sevenk love each other.       But recently,sevenk hea ...

  3. 三种做法:BZOJ 2780: [Spoj]8093 Sevenk Love Oimaster

    目录 题意 思路 AC_Code1 AC_Code2 AC_Code3 参考 @(bzoj 2780: [Spoj]8093 Sevenk Love Oimaster) 题意 链接:here 有\(n ...

  4. BZOJ 2780: [Spoj]8093 Sevenk Love Oimaster( 后缀数组 + 二分 + RMQ + 树状数组 )

    全部串起来做SA, 在按字典序排序的后缀中, 包含每个询问串必定是1段连续的区间, 对每个询问串s二分+RMQ求出包含s的区间. 然后就是求区间的不同的数的个数(经典问题), sort queries ...

  5. BZOJ.2780.[SPOJ8093]Sevenk Love Oimaster(广义后缀自动机)

    题目链接 \(Description\) 给定n个模式串,多次询问一个串在多少个模式串中出现过.(字符集为26个小写字母) \(Solution\) 对每个询问串进行匹配最终会达到一个节点,我们需要得 ...

  6. bzoj 2780 [Spoj]8093 Sevenk Love Oimaster

    LINK:Sevenk Love Oimaster 询问一个模式串在多少个文本串中出现过. 考虑广义SAM 统计这种数量问题一般有三种做法. 一种 暴力bitset 这道题可能可以过? 一种 暴力跳p ...

  7. 【刷题】BZOJ 2780 [Spoj]8093 Sevenk Love Oimaster

    Description Oimaster and sevenk love each other. But recently,sevenk heard that a girl named ChuYuXu ...

  8. bzoj 2780: [Spoj]8093 Sevenk Love Oimaster(广义SAM)

    题目大意:给出n个原串,再给出m个查询串.求每个查询串出现在了多少原串中. 题解 直接对原串建一个广义SAM,然后把每一个原串放到SAM上跑一跑,记录一下每一个状态属于多少个原串,用$size$表示. ...

  9. bzoj 2780: [Spoj]8093 Sevenk Love Oimaster【广义SAM】

    AC自动机比较简单,把询问串做成AC自动机然后模板串边跑变更新即可 SAM是把模板串做成广义SAM,然后每个节点存有几个模板串经过,具体方法是每次更新暴力向上跳直到有时间戳我不会证为什么时间复杂度是对 ...

随机推荐

  1. 将 HttpPostedFile 转换成 Image 或者 Bitmap

    代码如下: HttpFileCollection httpfiles = context.Request.Files; files = httpfiles[i]; Image im = Image.F ...

  2. EBS OAF开发中的错误/异常处理(ErrorHandling) (转)

    原文地址 EBS OAF开发中的错误/异常处理(ErrorHandling) EBS OAF开发中的错误/异常处理(ErrorHandling) (版权声明,本人原创或者翻译的文章如需转载,如转载用于 ...

  3. python中字符串连接的四种方式

    以下实例展示了join()的使用方法 #!/usr/bin/python str = "-"; seq = ("a", "b", " ...

  4. C#中使用Spire.docx操作Word文档

    使用docx一段时间之后,一些地方还是不方便,然后就尝试寻找一种更加简便的方法. 之前有尝试过使用Npoi操作word表格,但是太烦人了,随后放弃,然后发现免费版本的spire不错,并且在莫种程度上比 ...

  5. Maven 打jar包,pom文件配置

    以下是pom.xml文件的相关配置. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="h ...

  6. 难度2:ASCII码排序

    ASCII码排序 难度:2描述: 输入三个字符(可以重复)后,按各字符的ASCII码从小到大的顺序输出这三个字符. 输入: 第一行输入一个数N,表示有N组测试数据.后面的N行输入多组数据,每组输入数据 ...

  7. 用MyEclipse JPA创建项目(一)

    MyEclipse 3.15 Style——在线购买低至75折!火爆开抢>> [MyEclipse最新版下载] 本教程介绍了MyEclipse中的一些基于JPA的功能. 阅读本教程时,了解 ...

  8. 2018-2019-2 《网络对抗技术》Exp0 Kali安装 Week1 20165202

    2018-2019-2 <网络对抗技术>Exp0 Kali安装 Week1 20165202 一.下载 虚拟机环境 Kali所需要虚拟机软件为VMware,在之前其他课程中使用的即为VMw ...

  9. grafana dashboard的导入导出

    grafana的官方提供了很多社区或者官方设置的漂亮的dashboard,地址如下: 点击打开链接 导入图表大大节省了我们配置监控的时间,非常方便. 以linux host overview为例,首先 ...

  10. Java中多线程

    引 如果对什么是线程.什么是进程仍存有疑惑,请先Google之,因为这两个概念不在本文的范围之内. 用多线程只有一个目的,那就是更好的利用cpu的资源,因为所有的多线程代码都可以用单线程来实现.说这个 ...