【字符串处理】AC自动机知识点&代码
代码:
#include<iostream>
#include<vector>
#include<cstdio>
#include<queue>
#include<map>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<set>
#include<cstring>
using namespace std;
typedef long long ll;
const ll INF=;
const int MAXN=;
int trie[MAXN][]; //i到j的编号
int val[MAXN]; //代表这个节点有多少单词
int fail[MAXN]; //fail指针,意为失配时去的位置
int cnt;
queue<int> q;
void ins(string str){ //添加单词
int last=; //代表着多次失配之后最后跳到的地方
for(int i=;i<(int)str.length();i++){
int v=str[i]-'a';
if(!trie[last][v]) trie[last][v]=++cnt; //如果没有对应的这个节点的话,就加上(标记)
last=trie[last][v]; //将上次的编号记住
}
val[last]++; //单词数+1
}
void build(){
for(int i=;i<;i++) //遍历与根节点连接的点
if(trie[][i]){ //如果有这个字母的儿子
fail[trie[][i]]=; //将fail指针指向根
q.push(trie[][i]); //加入搜索队列 }
while(!q.empty()){
int u=q.front();
q.pop();
for(int i=;i<;i++){ //枚举其每个儿子
if(trie[u][i]){ //如果有这个节点
fail[trie[u][i]]=trie[fail[u]][i]; //fail指针指向父节点(当前节点)的fail指针指向的节点的相同字母节点
q.push(trie[u][i]); //加入这个点
}
else{
trie[u][i]=trie[fail[u]][i]; //没有这个点的话将其等同于父节点(当前节点)的fail指针的相同字母节点
}
}
}
}
int query(string str){
int last=,ans=;
for(int i=;i<(int)str.length();i++){
last=trie[last][str[i]-'a']; //获得当前字母,当前位置的编号
for(int j=last;j&&~val[j];j=fail[j]) { //只要没有结尾,就按照fail路线走
ans+=val[j]; //加上以这个节点结尾的单词书亮
val[j]=-; //已经拿走了,所以没有了
}
}
return ans;
}
int main()
{
ios_base::sync_with_stdio(false);
cin.tie();
int n;
cin>>n;
string st;
for(int i=;i<n;i++){
cin>>st;
ins(st); //添加单词
}
build(); //初始化
cin>>st;
cout<<query(st); //输出 return ;
}
查看神奇代码
0.容器
大部分人是用struct或者class实现内部的函数,但是作为考场上最倩的仔(雾,我决定使用数组存。
(很不理解为什么要用struct存,OI又不是写工程)
int trie[MAXN][27]; //代表从前、后的编号
int val[MAXN]; //代表这个节点有多少单词
int fail[MAXN]; //fail指针,意为失配时去的位置
int cnt; //当前编号
MAXN是数据的范围,是一个常量
1.插入
插入相当于从根一直走到最后一个字母的位置,没有就插入并赋予它一个新编号
void ins(string str){ //添加单词
int last=0; //代表着多次失配之后最后跳到的地方
for(int i=0;i<(int)str.length();i++){
int v=str[i]-'a';
if(!trie[last][v]) trie[last][v]=++cnt; //如果没有对应的这个节点的话,就加上(标记为新的节点,赋予其新编号)
last=trie[last][v]; //将这次的编号记住,以便下次使用
}
val[last]++; //单词数+1
}
2.Build
Build的是Fail指针:
先把所有与根节点连接的点的Fail指针指向根,然后将它们加入队列
然后BFS整棵树:
取出队首,然后将其儿子的Fail指针指向父亲的fail指针指向的相同字母的节点
赋予队首不存在的子节点父亲节点的Fail指针指向的相同字母的节点的编号
void build(){
for(int i=0;i<26;i++) //遍历与根节点连接的点
if(trie[0][i]){ //如果有这个字母的儿子
fail[trie[0][i]]=0; //将fail指针指向根
q.push(trie[0][i]); //加入搜索队列
}
while(!q.empty()){
int u=q.front();
q.pop();
for(int i=0;i<26;i++){ //枚举其每个儿子
if(trie[u][i]){ //如果有这个节点
fail[trie[u][i]]=trie[fail[u]][i]; //fail指针指向父节点(当前节点)的fail指针指向的节点的相同字母节点
q.push(trie[u][i]); //加入这个点
}
else{
trie[u][i]=trie[fail[u]][i]; //没有这个点的话将其等同于父节点(当前节点)的fail指针的相同字母节点
}
}
}
}
int query(string str){
int last=0,ans=0;
for(int i=0;i<(int)str.length();i++){
last=trie[last][str[i]-'a']; //获得当前字母,当前位置的编号
for(int j=last;j&&~val[j];j=fail[j]) { //只要没有结尾,就按照fail路线走
ans+=val[j]; //加上以这个节点结尾的单词书亮
val[j]=-1; //已经拿走了,所以没有了
}
}
return ans;
}
3.顺序
输入单词
ins
build
输出query
int n;
cin>>n;
string st;
for(int i=0;i<n;i++){
cin>>st;
ins(st); //添加单词
}
build(); //初始化
cin>>st;
cout<<query(st); //输出
本模版可以在洛谷P3808 https://www.luogu.org/problemnew/show/P3808提交,已通过
【字符串处理】AC自动机知识点&代码的更多相关文章
- 字符串处理-AC自动机
估计在OJ上刷过题的都会对AC自动机这个名词很感兴趣,同样,记得去年ACM暑期集训的时候,在最后讲到字符串部分,听说了这个算法的名字之后就对于它心向往之,AC正好是Accept的简称,字面意义上的理解 ...
- HDU-2222 Keywords Search 字符串问题 AC自动机
题目链接:https://cn.vjudge.net/problem/HDU-2222 题意 给一些关键词,和一个待查询的字符串 问这个字符串里包含多少种关键词 思路 AC自动机模版题咯 注意一般情况 ...
- 字符串(AC自动机):HDU 5129 Yong Zheng's Death
Yong Zheng's Death Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 512000/512000 K (Java/O ...
- 2017ACM暑期多校联合训练 - Team 8 1006 HDU 6138 Fleet of the Eternal Throne (字符串处理 AC自动机)
题目链接 Problem Description The Eternal Fleet was built many centuries ago before the time of Valkorion ...
- HDU-2896 病毒侵袭 字符串问题 AC自动机
题目链接:https://cn.vjudge.net/problem/HDU-2896 题意 中文题 给一些关键词和一个字符串,问字符串里包括了那几种关键词 思路 直接套模版 改insert方法,维护 ...
- HDU-3065 病毒侵袭持续中 字符串问题 AC自动机
题目链接:https://cn.vjudge.net/problem/HDU-3065 题意 跟上一道题是几乎一模一样,这次是统计关键词的出现次数 一个相当坑的地方,注意多组样例 思路 套模版 改in ...
- 字符串(AC自动机):COCI 2015 round 5 divljak
aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAy0AAANaCAIAAAALVTQoAAAgAElEQVR4nOy9X2hbx773PXfrQgQjDq
- 字符串:AC自动机
给出一个字典和一个模式串,问模式串中出现几个字典中的单词 最后一行是大串,之前输入的是小串 #include<iostream> #include<cstdio> using ...
- 多模字符串匹配算法之AC自动机—原理与实现
简介: 本文是博主自身对AC自动机的原理的一些理解和看法,主要以举例的方式讲解,同时又配以相应的图片.代码实现部分也予以明确的注释,希望给大家不一样的感受.AC自动机主要用于多模式字符串的匹配,本质上 ...
随机推荐
- CloseableHttpClient 在使用过程中遇到的问题
代码是前辈写的,在对代码进行压测的时候遇到了个问题,最大线程是 不能超过setDefaultMaxPerRoute设置的数字,一点超过 就会死掉.这里会报错 connection pool shut ...
- jQuery——stop
为什么要停止动画? 对同一个元素,如果拥有一个以上的动画对其加以作用,那么后面的动画会被放入一个动画队列中.动画队列的动画是在其上一个动画完成以后才会执行. 控制两个参数四种情况 1.第一个参数表示后 ...
- JS——模拟百度搜索
注意事项: 1.for循环移除子节点时,其长度是变化的 2.在文档流中,input.img.p等标签与其他标签有3px的距离,利用左浮动,可以消除3px距离 3.背景图片定位时,第一个值是x轴方向的值 ...
- Oracle Sequence不设置cache参数的几个潜在问题(转载)
转载于 http://www.uml.org.cn/sjjm/201204065.asp 在Oracle中,我们没有MYSQL和SQL Server ...
- C# 计算百分比
//计算比率 decimal A =(decimal) 200.20; decimal B = (decimal)1000.20; decimal t = decimal.Parse((A/B).To ...
- webstrom常用键
常用快捷键—Webstorm入门指南 提高代码编写效率,离不开快捷键的使用,Webstorm拥有丰富的代码快速编辑功能,你可以自由配置功能快捷键. 快捷键配置 点击“File”-> “setti ...
- linux强制踢出已登录的用户及本地用户
方法一: pkill -kill -t pts/0 方法二: fuser -k /dev/pts/0 你也可以给他发送关闭信息然后关闭 echo "你被管理员踢出了" > / ...
- js可以随意拖拽的div的实现
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 网络编程:tcp、udp、socket、struct、socketserver
一.TCP.UDP 一.ARP(Address Resolution Protocol)即地址解析协议,用于实现从 IP 地址到 MAC 地址的映射,即询问目标IP对应的MAC地址. 二.在网络通信中 ...
- Mac下对Android apk反编译
在Mac上进行反编译apk,需要三个工具,分别为(附下载地址): apktool,下载Mac版 作用:资源文件获取,能够提取出图片文件和布局文件进行使用查看 dex2jar,下载最新的即可,目前是2. ...