P3796 【模板】AC自动机
AC自动机的模板
简单的理解就是字典树上的KMP
注意数组不要开太大
不然每次memset耗时太多
有一个小优化
每次走 fail 边找匹配时只有一些会更新答案
那么就可以把没用的fail边压缩掉
设 g[x] 表示从 x 点一直走 fail 边,走到的第一个有结束标记的点
那么找匹配时就只有要 g 边
然后就是模板了
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<queue>
using namespace std;
const int N=5e4+;
int len,n;
char s[];
inline void read_s()//文本串比较大,可以用快读加速
{
char ch=getchar(); len=;
while(ch<'a'||ch>'z') ch=getchar();
while(ch>='a'&&ch<='z')
{
s[++len]=ch;
ch=getchar();
}
}
int c[N][],pd[N],fail[N],g[N],cnt;
char a[][]; int l[];
inline void clr()//清空AC自动机
{
memset(c,,sizeof(c));
memset(pd,,sizeof(pd));
memset(fail,,sizeof(fail));
memset(g,,sizeof(g));
cnt=;
}
inline void ins(int num)//插入单词到自动机(跟字典树没区别)
{
int u=;
for(int i=;i<l[num];i++)
{
int v=a[num][i]-'a'+;
if(!c[u][v]) c[u][v]=++cnt;
u=c[u][v];
}
pd[u]=num;
}
queue <int> q;
inline void pre()//预处理fail和g
{
for(int i=;i<=;i++)
if(c[][i]) q.push(c[][i]);
while(!q.empty())
{
int u=q.front(); q.pop();
for(int i=;i<=;i++)
{
int v=c[u][i];
if(!v) c[u][i]=c[fail[u]][i];//失配了直接走失配边
else
{
fail[v]=c[fail[u]][i];
g[v]= pd[fail[v]] ? fail[v] : g[fail[v]];//处理g
q.push(v);
}
}
}
}
int mx,ans[];
inline void query()//AC自动机匹配(好像跟字典树也没什么区别)
{
memset(ans,,sizeof(ans));
int u=;
for(int i=;i<=len;i++)
{
u=c[u][s[i]-'a'+];
if(pd[u]) ans[pd[u]]++;
for(int j=g[u];j;j=g[j])
ans[pd[j]]++;
//计算匹配
}
mx=;
for(int i=;i<=n;i++)
mx=max(mx,ans[i]);
cout<<mx<<endl;
for(int i=;i<=n;i++)
if(ans[i]==mx)
printf("%s\n",a[i]);
}
int main()
{
while()
{
scanf("%d",&n); if(!n) break;
clr();
for(int i=;i<=n;i++)
{
scanf("%s",a[i]);
l[i]=strlen(a[i]);
ins(i);
}
pre();
read_s();
query();
}
return ;
}
P3796 【模板】AC自动机的更多相关文章
- luoguP3808[模板]AC自动机(简单版)
传送门 ac自动机模板题,裸的多串匹配 代码: #include<cstdio> #include<iostream> #include<algorithm> #i ...
- luoguP3796[模板]AC自动机(加强版)
传送门 ac自动机模板,可能我写的ac自动机是有点问题的,所以跑的有些慢 暴力跳fail统计 代码: #include<cstdio> #include<iostream> # ...
- 算法模板——AC自动机
实现功能——输入N,M,提供一个共计N个单词的词典,然后在最后输入的M个字符串中进行多串匹配(关于AC自动机算法,此处不再赘述,详见:Aho-Corasick 多模式匹配算法.AC自动机详解.考虑到有 ...
- 模板 AC自动机
题目描述 有$N$ 个由小写字母组成的模式串以及一个文本串$T$ .每个模式串可能会在文本串中出现多次.你需要找出哪些模式串在文本串$T$ 中出现的次数最多. 输入输出格式 输入格式: 输入含多组数据 ...
- 算法竞赛模板 AC自动机
AC自动机基本操作 (1) 在AC自动机中,我们首先将每一个模式串插入到Trie树中去,建立一棵Trie树,然后构建fail指针. (2) fail指针,是穿插在Trie树中各个结点之间的指针,顾名思 ...
- 洛谷.3808/3796.[模板]AC自动机
题目链接:简单版,增强版 简单版: #include <cstdio> #include <cstring> const int N=1e6+5,S=26; char s[N] ...
- 模板—AC自动机
#include<iostream> #include<cstdio> #include<cstring> using namespace std; struct ...
- 模板——AC自动机
传送门:QAQQAQ 定义nxt[u]=v表示从u开始不断沿着失配边跳到的第一个是标记点的端点v,那么我们再匹配时沿着last跳,每跳到一个last,它就一定对应一个模式串,所以效率是非常高的. 和K ...
- AC自动机例题
P3808 [模板]AC自动机(简单版) [题目描述] 给定n个模式串和1个文本串,求有多少个模式串在文本串里出现过. #include<bits/stdc++.h> using name ...
- 「kuangbin带你飞」专题十七 AC自动机
layout: post title: 「kuangbin带你飞」专题十七 AC自动机 author: "luowentaoaa" catalog: true tags: - ku ...
随机推荐
- 基于:Hadoop 2.6.0-cdh5.4.0 hive1.1.0 HBase 1.0.0-cdh5.4.0 关键配置文件
core-site.xml <configuration> <property> <name>fs.defaultFS</name> <value ...
- saltstact的安装与配置
Saltstack是一个服务器基础架构集中化管理平台,具备配置管理.远程执行.监控等功能,人们一般习惯把saltstack比作成简化版的puppet和加强版的func.saltstack基于Pytho ...
- Windows cmd 将命令(/指令)写到一个文件里,直接运行这个文件。提高工作效率
Windows cmd 批处理(cmd/bat)文件的简单使用介绍 前言 如果你想我一样,要每天都需要在cmd上,用键盘去敲击相同的命令,时间一长,你就觉得很无聊.有没有什么比较高效的方法,让我们不用 ...
- 线程池的原理以及实现线程池的类ExecutorService中方法的使用
1.线程池:线程池就是就像一个容器,而这个容器就是用来存放线程的,且有固定的容量. 如果没有线程池,当需要一个线程来执行任务时就需要创建一个线程,我们设创建线程的时间为t1,执行线程的时间为t2,销毁 ...
- java中什么是代码点,什么是代码单元?
1.代码点&代码单元,是从Unicode标准而来的术语,Unicode标准的核心是一个编码字符集,它为每一个字符分配一个唯一数字.Unicode标准始终使用16进制数字,并且在书写时在前面加上 ...
- 3、PACBIO下机数据如何看
转载:http://www.cnblogs.com/jinhh/p/8328818.html 三代测序的下机数据都有哪些,以及他们具体的格式是怎么样的(以sequel 平台为主). 测序过程 SMRT ...
- Luogu 2824 [HEOI2016/TJOI2016]排序
BZOJ 4552 挺妙的解法. 听说这题直接用一个桶能拿到$80 \ pts$ 发现如果是一个排列的话,要对这个序列排序并不好做,但是假如是$01$序列的话,要对一个区间排序还是很简单的. 发现最后 ...
- Luogu 2939 [USACO09FEB]改造路Revamping Trails && Luogu 4568 [JLOI2011]飞行路线
双倍经验 写这两题之前被大佬剧透了呜呜呜. 分层图+最短路. 因为有$k$次机会能够把路径的费用变为$0$,我们可以建$k + 1$层图,对于每一层图我们把原来的边权和双向边连到上面去,而对于层与层之 ...
- Jsp入门第一天
1. JSP: 1). WHY: JSP 是简 Servlet 编写的一种技术, 它将 Java 代码和 HTML 语句混合在同一个文件中编写,只对网页中的要动态产生的内容采用 Java 代码来编写, ...
- eclipse workspace 共享设置
总结一下,复制工作空间配置步骤如下: 1 使用eclipse新建workspace. 2 将新建的workspace下的.metadata\.plugins内容全部删除. 3 将需要拷贝的worksp ...