题目:P3808:https://www.luogu.org/problemnew/show/P3808

P3796:https://www.luogu.org/problemnew/show/P3796

从这里学了下AC自动机:http://www.cnblogs.com/cjyyb/p/7196308.html

我的理解大概就是构建一棵由模式串组成的 Trie 树,然后把文本串一节一节放在上面查找;

失配指针指向的是结尾字母和自己一样的、Trie 树上的其他分支,大约就是在找后缀这样的感觉;

所以文本串每加入一个字符,就在 Trie 树上查找以这个字符结尾的后缀模式串,所以能找到所有出现过的;

慕名已久的AC自动机原来也挺简单的嘛!

代码如下:

P3808:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
int const maxn=1e6+;
int n,cnt;
queue<int>q;
struct N{
int fail,son[],end;
}AC[maxn];
void build(string s)
{
int l=s.length();
int nw=;
for(int i=;i<l;i++)
{
if(AC[nw].son[s[i]-'a']==)AC[nw].son[s[i]-'a']=++cnt;
nw=AC[nw].son[s[i]-'a'];
}
AC[nw].end++;
}
void get_fail()
{
AC[].fail=;
for(int i=;i<;i++)
{
if(AC[].son[i]==)continue;
AC[AC[].son[i]].fail=;
q.push(AC[].son[i]);
}
while(q.size())
{
int x=q.front(); q.pop();
for(int i=;i<;i++)
{
if(AC[x].son[i])
{
AC[AC[x].son[i]].fail=AC[AC[x].fail].son[i];
q.push(AC[x].son[i]);
}
else AC[x].son[i]=AC[AC[x].fail].son[i];
}
}
}
int query(string s)
{
int l=s.length();
int ret=,nw=;
for(int i=;i<l;i++)
{
nw=AC[nw].son[s[i]-'a'];
for(int t=nw;t&&AC[t].end!=-;t=AC[t].fail)
{
ret+=AC[t].end;
AC[t].end=-;
}
}
return ret;
}
int main()
{
scanf("%d",&n);
string s;
for(int i=;i<=n;i++)
{
cin>>s;
build(s);
}
get_fail();
cin>>s;
printf("%d\n",query(s));
return ;
}

P3796:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
int const maxn=1e6+;
int n,cnt;
queue<int>q;
string s[];
struct N{
int son[],fail,end;
}AC[maxn];
struct P{int num,pos;}ans[];
bool cmp(P x,P y)
{
if(x.num==y.num)return x.pos<y.pos;
else return x.num>y.num;
}
void clear(int x)
{
memset(AC[x].son,,sizeof AC[x].son);
AC[x].fail=; AC[x].end=;
}
void build(string s,int num)
{
int l=s.length();
int nw=;
for(int i=;i<l;i++)
{
if(!AC[nw].son[s[i]-'a'])AC[nw].son[s[i]-'a']=++cnt,clear(cnt);
nw=AC[nw].son[s[i]-'a'];
}
AC[nw].end=num;
}
void get_fail()
{
while(q.size())q.pop();
AC[].fail=;
for(int i=;i<;i++)
{
if(AC[].son[i]==)continue;
AC[AC[].son[i]].fail=; q.push(AC[].son[i]);
}
while(q.size())
{
int x=q.front(); q.pop();
for(int i=;i<;i++)
{
if(AC[x].son[i])
{
AC[AC[x].son[i]].fail=AC[AC[x].fail].son[i];
q.push(AC[x].son[i]);
}
else AC[x].son[i]=AC[AC[x].fail].son[i];
}
}
}
void query(string s)
{
int l=s.length();
int nw=;
for(int i=;i<l;i++)
{
nw=AC[nw].son[s[i]-'a'];
for(int t=nw;t/*&&t.end!=-1*/;t=AC[t].fail)
{
ans[AC[t].end].num++;
// AC[t].end=-1;
}
}
}
int main()
{
while()
{
scanf("%d",&n);
if(!n)return ;
cnt=; clear();
for(int i=;i<=n;i++)
{
cin>>s[i];
build(s[i],i);
ans[i].pos=i;
ans[i].num=;//
}
get_fail();
cin>>s[];
query(s[]);
sort(ans+,ans+n+,cmp);
printf("%d\n",ans[].num);
cout<<s[ans[].pos]<<endl;
for(int i=;i<=n;i++)
{
if(ans[i].num==ans[i-].num)
cout<<s[ans[i].pos]<<endl;
else break;
}
}
}

洛谷P3808 & P3796 AC自动机模板的更多相关文章

  1. 洛谷 - P3966 - 单词 - AC自动机

    https://www.luogu.org/problemnew/show/P3966 因为文本串就是字典本身,所以这个和平时的AC自动机不太一样.平时的query要沿着fail树把子树的出现次数依次 ...

  2. 洛谷.3121.审查(AC自动机 链表)

    题目链接 //删掉一个单词需要前移一段位置,用链表维护就好了 复杂度O(sum(len)) #include <cstdio> #include <cstring> #defi ...

  3. 洛谷 - P2444 - 病毒 - AC自动机

    https://www.luogu.org/problemnew/show/P2444 有点恶心,不太明白fail的意义. #include<bits/stdc++.h> using na ...

  4. 洛谷P3808 【模板】AC自动机(简单版)

    题目背景 这是一道简单的AC自动机模板题. 用于检测正确性以及算法常数. 为了防止卡OJ,在保证正确的基础上只有两组数据,请不要恶意提交. 管理员提示:本题数据内有重复的单词,且重复单词应该计算多次, ...

  5. 【刷题】洛谷 P3808 【模板】AC自动机(简单版)

    题目背景 这是一道简单的AC自动机模板题. 用于检测正确性以及算法常数. 为了防止卡OJ,在保证正确的基础上只有两组数据,请不要恶意提交. 管理员提示:本题数据内有重复的单词,且重复单词应该计算多次, ...

  6. 浅谈AC自动机模板

    什么是AC自动机? 百度百科 Aho-Corasick automaton,该算法在1975年产生于贝尔实验室,是著名的多模匹配算法. 要学会AC自动机,我们必须知道什么是Trie,也就是字典树.Tr ...

  7. (模板)AC自动机模板

    模板1. 给出模式串和文本串,文本串长度小于1e6,模式串长度之和小于1e6,求文本串中有多少模式串出现. 题目链接:https://www.luogu.org/problem/P3808 AC co ...

  8. HDU 2222 AC自动机模板题

    题目: http://acm.hdu.edu.cn/showproblem.php?pid=2222 AC自动机模板题 我现在对AC自动机的理解还一般,就贴一下我参考学习的两篇博客的链接: http: ...

  9. Match:Keywords Search(AC自动机模板)(HDU 2222)

    多模匹配 题目大意:给定很多个字串A,B,C,D,E....,然后再给你目标串str字串,看目标串中出现多少个给定的字串. 经典AC自动机模板题,不多说. #include <iostream& ...

随机推荐

  1. 基于openstack平台的几种Cloud DB解决方案

    方案一.openstack 官方 trove解决方案 此方案进行过镜像的打包,由于网络问题,还未能成功实现 方案二.salt 或者ansible+ docker 由于 docker部署数据库,在数据库 ...

  2. JS高级——逻辑中断

    1.表达式1||表达式2:表达式1为真,返回表达式1:表达式1为假,返回表达式2 2.表达式1&&表达2:表达式1为真,返回表达式2:表达式1为假,返回表达式1

  3. docker安装后出现Cannot connect to the Docker daemon

    启动docker service docker start docker安装后出现Cannot connect to the Docker daemon You need to add user in ...

  4. Day 11 文件和异常

    文件和异常 在实际开发中,常常需要对程序中的数据进行持久化操作,而实现数据持久化最直接简单的方式就是将数据保存到文件中.说到“文件”这个词,可能需要先科普一下关于文件系统的知识,对于这个概念,维基百科 ...

  5. 类 Fabric 主机管理程序开发

    类 Fabric 主机管理程序开发:1. 运行程序列出主机组或者主机列表2. 选择指定主机或主机组3. 选择让主机或者主机组执行命令或者向其传输文件(上传/下载4. 充分使用多线程或多进程5. 不同主 ...

  6. hdu 4876

    ZCC loves cards Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

  7. hstack()与vstack()函数

    ref: https://blog.csdn.net/csdn15698845876/article/details/73380803 1. hstack()函数 a,b只有一个维度:对第一个维度拼接 ...

  8. Problem 34

    Problem 34 https://projecteuler.net/problem=34 145 is a curious number, as 1! + 4! + 5! = 1 + 24 + 1 ...

  9. git 的简单使用(4)

    多人协作的工作模式通常是这样: 首先,可以试图用git push origin <branch-name>推送自己的修改: 如果推送失败,则因为远程分支比你的本地更新,需要先用git pu ...

  10. homework week 1

    第一周的作业 首先来完成第二个作业, 编写登录接口, 因为视频上并没有相关的教程, 就在网上搜了一下读写文件的语句, 粗略了解. f1 = open("data.txt",&quo ...