传送门

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自动机的更多相关文章

  1. luoguP3808[模板]AC自动机(简单版)

    传送门 ac自动机模板题,裸的多串匹配 代码: #include<cstdio> #include<iostream> #include<algorithm> #i ...

  2. luoguP3796[模板]AC自动机(加强版)

    传送门 ac自动机模板,可能我写的ac自动机是有点问题的,所以跑的有些慢 暴力跳fail统计 代码: #include<cstdio> #include<iostream> # ...

  3. 算法模板——AC自动机

    实现功能——输入N,M,提供一个共计N个单词的词典,然后在最后输入的M个字符串中进行多串匹配(关于AC自动机算法,此处不再赘述,详见:Aho-Corasick 多模式匹配算法.AC自动机详解.考虑到有 ...

  4. 模板 AC自动机

    题目描述 有$N$ 个由小写字母组成的模式串以及一个文本串$T$ .每个模式串可能会在文本串中出现多次.你需要找出哪些模式串在文本串$T$ 中出现的次数最多. 输入输出格式 输入格式: 输入含多组数据 ...

  5. 算法竞赛模板 AC自动机

    AC自动机基本操作 (1) 在AC自动机中,我们首先将每一个模式串插入到Trie树中去,建立一棵Trie树,然后构建fail指针. (2) fail指针,是穿插在Trie树中各个结点之间的指针,顾名思 ...

  6. 洛谷.3808/3796.[模板]AC自动机

    题目链接:简单版,增强版 简单版: #include <cstdio> #include <cstring> const int N=1e6+5,S=26; char s[N] ...

  7. 模板—AC自动机

    #include<iostream> #include<cstdio> #include<cstring> using namespace std; struct ...

  8. 模板——AC自动机

    传送门:QAQQAQ 定义nxt[u]=v表示从u开始不断沿着失配边跳到的第一个是标记点的端点v,那么我们再匹配时沿着last跳,每跳到一个last,它就一定对应一个模式串,所以效率是非常高的. 和K ...

  9. AC自动机例题

    P3808 [模板]AC自动机(简单版) [题目描述] 给定n个模式串和1个文本串,求有多少个模式串在文本串里出现过. #include<bits/stdc++.h> using name ...

  10. 「kuangbin带你飞」专题十七 AC自动机

    layout: post title: 「kuangbin带你飞」专题十七 AC自动机 author: "luowentaoaa" catalog: true tags: - ku ...

随机推荐

  1. hadoop-2.7.3.tar.gz + spark-2.0.2-bin-hadoop2.7.tgz + zeppelin-0.6.2-incubating-bin-all.tgz(master、slave1和slave2)(博主推荐)(图文详解)

    不多说,直接上干货! 我这里,采取的是ubuntu 16.04系统,当然大家也可以在CentOS6.5里,这些都是小事 CentOS 6.5的安装详解 hadoop-2.6.0.tar.gz + sp ...

  2. TCP/IP 笔记 6 netstat -s 命令查看每个协议统计数据

    netstat -s 命令,查看每个协议统计数据的常用方法 lenovo-myc@lenovomyc-Lenovo-Product:~$ netstat -s Ip: total packets re ...

  3. saltstact的安装与配置

    Saltstack是一个服务器基础架构集中化管理平台,具备配置管理.远程执行.监控等功能,人们一般习惯把saltstack比作成简化版的puppet和加强版的func.saltstack基于Pytho ...

  4. iOS UITableView制作类似QQ好友列表视图

                #import <UIKit/UIKit.h> @interface AppDelegate : UIResponder <UIApplicationDele ...

  5. 【摘自张宴的"实战:Nginx"】nginx配置

    user nobody;worker_processes 2; #error_log logs/error.log;error_log logs/error.log notice;#error_log ...

  6. C#调用C++类库的几种方式

    1.  直接调用C++类库中的公共方法 使用DllImport特性对方法进行调用,比如一个C++类库SampleCppWrapper.dll中的公共方法: extern "C" _ ...

  7. C语言-郝斌笔记-001求二次方程的根

    求二次方程的根 #include <stdio.h > #include<math.h> int main(void) { //把三个系数保存到计算机中 ; //=不表示相等, ...

  8. virtualbox复制虚拟机网络问题

    virtulbox复制虚拟机由于mac地址问题会导致网卡不可以用 1:修改mac地址 需要在virtualbox修改虚拟机网络选项卡下面的mac地址 2:修改ifcfg-eth0 把HWADDR的值设 ...

  9. emr问题处理

    --通过his病历号查询emr中对应的患者ID --通过患者ID找出患者所有的病历集合ID --通过病历集合查找患者所有的病历 --通过病历dataID查找对应的病历数据,病历存在大字段中 '; -- ...

  10. AttributeError: 'module' object has no attribute 'gfile'

    While running TensorFlow's classify_image, getting AttributeError: 'module' object has no attribute ...