洛谷 P3808 【模板】AC自动机(简单版) 题解
前置知识:
字典树。(会 \(\texttt{KMP}\) 就更好)
显然呢,本题用 字典树 和 \(\texttt{KMP}\) 无法解决问题。
所以我们发明了一个东西: \(\texttt{AC}\) 自动机!
自动AC就算了吧
首先,我们给这些串建字典树。
建完之后,我们求 失配指针 。
这是干嘛的?求完再说。
它表示以 \(i\) 节点为 结尾 的串的 后缀 有最大公共长度的 前缀的 结尾 编号.
可能有点绕,但是字符串匹配算法,一开始就是雾里云里,后来就是拨云见雾。
引用一张洛谷题解上的图吧。

然后,比方说第 \(3\) 层的 \(c\).
首先,以它结尾的后缀有: \(\texttt{c}\),\(\texttt{bc}\),\(\texttt{abc}\).
显然,从根开始的前缀(不含根)找不到 \(\texttt{abc}\).
但是,我们找到了 \(\texttt{bc}\).
所以,就指向了 \(\texttt{bc}\) 中的 结尾 编号的 \(c\) 的位置。
余下同理,读者自行推理。
下面,我们假设这个图的只有每个叶子节点都是一个单词的末尾。那么,假设我们要找在 \(\texttt{abcde}\) 中的次数,一开始 \(ans = 0\).流程为:
首先一路往下,到 \(d\) 之后发现 \(e\) 没了。这时 \(ans \gets ans+1\),即 \(ans = 1\).
这时我们创建一个空的 \(e\). 并且,对每个 \(e\) 也求一下 失配指针 :

然后,你走到了第二叉的“空节点” \(e\).
然后,你发现:由于失配指针的性质, \(e\) 上面这一段肯定在 \(\texttt{abcd}\) 中出现过(因为后缀和前缀匹配,而长度递减),所以也在原串中出现过。
然后, \(ans \gets ans+1\),即 \(ans = 2\).
接着,你又走到第三叉的“空节点” \(e\).
同样的道理,\(ans \gets ans+1\),即 \(ans = 3\).
接着,你发现当前的 \(e\) 指向根,于是迫不及待地走向了根。
然后你发现当前节点编号是 \(0\),结束。
\(ans = 3\),没有一点毛病,不得不承认这个算法很妙。
可是怎么求 \(\texttt{Fail}\) (失配指针) 呢?
显然,如果父亲节点有了失配指针,你只需比较 你自己 和 父亲失配指针的那一位 ,相同则指过去,不然呢就指根。
这是因为,父亲节点以上全部匹配,如果你自己也匹配就完事了;否则呢,就不匹配了。
你会发现,第 \(i\) 层的所有指针需要 \(i-1\) 层。所以宽搜!
时间复杂度:\(O(n)\).(常数较大,需要提高效率)
实际得分:\(100pts\).
#pragma GCC optimize(2)
#include<bits/stdc++.h>
using namespace std;
const int N=1e6+1;
inline int read(){char ch=getchar();int f=1;while(ch<'0' || ch>'9') {if(ch=='-') f=-f; ch=getchar();}
int x=0;while(ch>='0' && ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();return x*f;}
struct tree {
int fail,end; //失配指针,单词个数
int nxt[26];
};
tree t[N];
int cnt=0;
inline void build_tree(string s) {
int p=0;
for(int i=0,tt;i<s.length();i++) {
tt=s[i]-'a';
if(!t[p].nxt[tt]) t[p].nxt[tt]=++cnt;
p=t[p].nxt[tt];
} t[p].end++;
} //建树
queue<int>q;
inline void getFail() {
for(int i=0;i<26;i++)
if(t[0].nxt[i]) {
t[t[0].nxt[i]].fail=0;
q.push(t[0].nxt[i]);
} //根节点的儿子直接标记
while(!q.empty()) {
int now=q.front(); q.pop();
for(int i=0;i<26;i++)
if(t[now].nxt[i]) {
t[t[now].nxt[i]].fail=t[t[now].fail].nxt[i];
q.push(t[now].nxt[i]);
} else t[now].nxt[i]=t[t[now].fail].nxt[i];
} //宽搜
}
inline int AC(string s) {
int p=0,ans=0;
for(int i=0,tt;i<s.length();i++) {
tt=s[i]-'a'; p=t[p].nxt[tt];
for(int j=p;j && t[j].end!=-1;j=t[j].fail) {
ans+=t[j].end;
t[j].end=-1; //为了防止一个子树被走多次
} //只要不为空,就一直记录
} return ans;
}
int main(){
int n=read(); string s;
while(n--) {
cin>>s;
build_tree(s);
} t[0].fail=0; //初始化
getFail();
cin>>s; int x=AC(s);
printf("%d\n",x);
return 0;
}
洛谷 P3808 【模板】AC自动机(简单版) 题解的更多相关文章
- 洛谷P3808 & P3796 AC自动机模板
题目:P3808:https://www.luogu.org/problemnew/show/P3808 P3796:https://www.luogu.org/problemnew/show/P37 ...
- 洛谷 - P3966 - 单词 - AC自动机
https://www.luogu.org/problemnew/show/P3966 因为文本串就是字典本身,所以这个和平时的AC自动机不太一样.平时的query要沿着fail树把子树的出现次数依次 ...
- [模板][P3808]AC自动机(简单版)
Description: 求n个模式串中有几个在文本串中出现 Solution: 模板,详见代码: #include<bits/stdc++.h> using namespace std; ...
- 洛谷.3121.审查(AC自动机 链表)
题目链接 //删掉一个单词需要前移一段位置,用链表维护就好了 复杂度O(sum(len)) #include <cstdio> #include <cstring> #defi ...
- 洛谷 - P2444 - 病毒 - AC自动机
https://www.luogu.org/problemnew/show/P2444 有点恶心,不太明白fail的意义. #include<bits/stdc++.h> using na ...
- 洛谷 P3804 [模板] 后缀自动机
题目:https://www.luogu.org/problemnew/show/P3804 模仿了一篇题解,感觉很好写啊. 代码如下: #include<cstdio> #include ...
- 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)
To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...
- 洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心)
洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/132 ...
- 洛谷P3808 【模板】AC自动机(简单版)
题目背景 这是一道简单的AC自动机模板题. 用于检测正确性以及算法常数. 为了防止卡OJ,在保证正确的基础上只有两组数据,请不要恶意提交. 管理员提示:本题数据内有重复的单词,且重复单词应该计算多次, ...
- 【刷题】洛谷 P3808 【模板】AC自动机(简单版)
题目背景 这是一道简单的AC自动机模板题. 用于检测正确性以及算法常数. 为了防止卡OJ,在保证正确的基础上只有两组数据,请不要恶意提交. 管理员提示:本题数据内有重复的单词,且重复单词应该计算多次, ...
随机推荐
- Ubuntu 16.04 PXE+kickstart部署系统
#PXE+TFTP+Kickstart 自动部署服务器系统系统Ubuntu16.04apt-get install isc-dhcp-servervim /etc/default/isc-dhcp-s ...
- Ta说:2016微软亚洲研究院第二届博士生论坛
"聚合多元人才创造无尽可能,让每一位优秀博士生得到发声成长机会"可以说是这次微软亚洲研究院博士生论坛最好的归纳了.自去年首次举办以来,这项旨在助力青年研究者成长的项目迅速得到了 ...
- Win10+WSL2+Ubuntu 18.04(WSL下)+VS Code(Win10下)+TexLive 2019(Ubuntu下)安装和配置
本人手头电脑是Win10 Home版全新安装的系统,由于不想在新系统盘里面安装TexLive导致固态硬盘不断扩大,所以,考虑安装Ubuntu做为WSL,然后把TexLive安装在Ubuntu,并通过V ...
- 远程终端协议 TELNET
远程终端协议 TELNET 1.1.概述 TELNET 是一个简单的远程终端协议,也是因特网的正式标准. 用户用 TELNET 就可在其所在地通过 TCP 连接的23端口,使用主机名或 IP 地址登录 ...
- phpstudy渗透到服务器
0x00 目标站点www.test.ichunqiu 0x01 尝试登陆系统 -尝试弱密码登陆 结果:forbidden!!! -尝试万能账号密码登陆 1‘ or 1=1--+ 和 1‘ or 1=1 ...
- 前端每日实战:60# 视频演示如何用纯 CSS 创作一块乐高积木
效果预览 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/qKKqrv 可交互视频 此视频是可 ...
- Django进行数据迁移时,报错:(1064, "You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '(6) NOT NULL)' at line 1")
进行数据迁移时: 第一步: 命令:python manage.py makemigrations 在对应的应用里面的migrations文件夹中产生了一个0001_initial.py文件 第二步:执 ...
- Python基础-检测密码,一些网站会给密码强加一些规则。
输入一个字符串,检测是否是合法的密码:1)密码必须包含8个字符2)密码只能包含英文字母和数字3)密码至少包含两个数字 首先我讲一下用到的方法 s为字符串 len(s) 求出字符串的长度. list(s ...
- 峰哥说技术: 05-Spring Boot条件注解注解
Spring Boot深度课程系列 峰哥说技术—2020庚子年重磅推出.战胜病毒.我们在行动 05 峰哥说技术 Spring Boot条件注解 @EnableAutoConfiguration开启自 ...
- JavaScript的函数(一)
,1,在javascript中,函数即对象.函数里面的参数可以是个函数,例如: data.sort(function(a,b){return a-b;}) 函数的返回值,return语句导致函数停止执 ...