【字符串处理】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自动机主要用于多模式字符串的匹配,本质上 ...
随机推荐
- Java—解压zip文件
import java.io.BufferedOutputStream; import java.io.File; import java.io.FileOutputStream; import ja ...
- (转)在 vue-cli 脚手架中引用 jQuery、bootstrap 以及使用 sass、less 编写 css [vue-cli配置入门]
写在前面: 本文是vue-手摸手教你使用vue-cli脚手架-详细步骤图文解析之后,又一篇关于vue-cli脚手架配置相关的文章,因为有些文章步骤不够清晰,当时我引入JQuery.bootstrap的 ...
- Unity引擎 UGUI
Unity UGUI讲解 1.导入UI图片资源 2.设置参数: TextureType(纹理类型) 精灵 2D and UI SpriteMode(精灵模式) Single(单) multiple( ...
- Leetcode0006--ZigZag Conversion
[转载请注明]https://www.cnblogs.com/igoslly/p/9017638.html 来看一下题目: The string "PAYPALISHIRING" ...
- SQL Server建库-建表-建约束
----------------------------------------SQL Server建库-建表-建约束创建School数据库------------------------------ ...
- FTP工作原理
FTP工作原理 FTP两种传输方式:1.ASCII传输2.二进制传输 FTP主被动原理: 主动方式:1.用户与服务器建立控制通道2.客户端发出PORT指令,主动告诉服务器端口号3.服务器主动通过20端 ...
- 轻松理解 Android Binder,只需要读这一篇
在 Android 系统中,Binder 起着非常重要的作用,它是整个系统 IPC 的基石.网上已经有很多文章讲述 Binder 的原理,有的讲的比较浅显,没有触及到关键,有的讲的太过于深入底层,难以 ...
- Redis 之持久化(rdb、aof)
Redis的持久化有2种方式 1快照 2是日志 测试aof:
- scp: /xxxx: not a regular file
问题描述 scp root@10.2.1.92:/home /home/wangju/databakroot@10.2.1.92's password: xxxxscp: /home: not a r ...
- JNI数组操作
在Java中数组分为两种: 1.基本类型数组 2.对象类型(Object[])的数组(数组中存放的是指向Java对象中的引用) 一个能通用于两种不同类型数组的函数: GetArrayLength(ja ...