原题链接

前置知识:

字典树。(会 \(\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自动机(简单版) 题解的更多相关文章

  1. 洛谷P3808 & P3796 AC自动机模板

    题目:P3808:https://www.luogu.org/problemnew/show/P3808 P3796:https://www.luogu.org/problemnew/show/P37 ...

  2. 洛谷 - P3966 - 单词 - AC自动机

    https://www.luogu.org/problemnew/show/P3966 因为文本串就是字典本身,所以这个和平时的AC自动机不太一样.平时的query要沿着fail树把子树的出现次数依次 ...

  3. [模板][P3808]AC自动机(简单版)

    Description: 求n个模式串中有几个在文本串中出现 Solution: 模板,详见代码: #include<bits/stdc++.h> using namespace std; ...

  4. 洛谷.3121.审查(AC自动机 链表)

    题目链接 //删掉一个单词需要前移一段位置,用链表维护就好了 复杂度O(sum(len)) #include <cstdio> #include <cstring> #defi ...

  5. 洛谷 - P2444 - 病毒 - AC自动机

    https://www.luogu.org/problemnew/show/P2444 有点恶心,不太明白fail的意义. #include<bits/stdc++.h> using na ...

  6. 洛谷 P3804 [模板] 后缀自动机

    题目:https://www.luogu.org/problemnew/show/P3804 模仿了一篇题解,感觉很好写啊. 代码如下: #include<cstdio> #include ...

  7. 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)

    To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...

  8. 洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心)

    洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/132 ...

  9. 洛谷P3808 【模板】AC自动机(简单版)

    题目背景 这是一道简单的AC自动机模板题. 用于检测正确性以及算法常数. 为了防止卡OJ,在保证正确的基础上只有两组数据,请不要恶意提交. 管理员提示:本题数据内有重复的单词,且重复单词应该计算多次, ...

  10. 【刷题】洛谷 P3808 【模板】AC自动机(简单版)

    题目背景 这是一道简单的AC自动机模板题. 用于检测正确性以及算法常数. 为了防止卡OJ,在保证正确的基础上只有两组数据,请不要恶意提交. 管理员提示:本题数据内有重复的单词,且重复单词应该计算多次, ...

随机推荐

  1. hiho一下:Beautiful String

    hiho一下:Beautiful String 记不清这是 hiho一下第几周的题目了,题目不难,不过对于练习编程,训练思维很有帮助.况且当时笔者处于学习算法的早期, 所以也希望刚接触算法的同学能多去 ...

  2. 我的学习归纳方法(以学习Maven为例)

    以我的个人角度来看待学习这件长久的事,希望对你有帮助,也希望你能提一下你的意见 本文初衷 把自己模板化 以此篇为引,与同行沟通心得,所以在此严重要求如果你有对应的心得还请能回复下,真心感谢!(鞠躬) ...

  3. 【转载】java 获取路径的各种方法

    转载只供个人学习参考,查看请前往原出处:http://www.cnblogs.com/guoyuqiangf8/p/3506768.html 主要方法有: (1).request.getRealPat ...

  4. 用pyqt5简单实现Mqtt调试助手-初学python

    最近在学习pyqt5,因为我们是做远程物联网设备的,所以就做个mqtt调试助手来练手.第一次做这种程序,没有加异常处理,会有很多不足,欢迎留言拍砖,直接上代码了 这个是程序入口, # 使用前先安装py ...

  5. jQuery样式及html属性操作

    样式及html属性操作1,行内样式 css(); a:获取样式 元素.css(样式名称); b:设置单个样式 元素.css("样式名称":"样式值"); c:设 ...

  6. CSV文件注入漏洞简析

    “对于网络安全来说,一切的外部输入均是不可信的”.但是CSV文件注入漏洞确时常被疏忽,究其原因,可能是因为我们脑海里的第一印象是把CSV文件当作了普通的文本文件,未能引起警惕. 一.漏洞定义 攻击者通 ...

  7. iOS开发日常笔记01

    为什么有initWithCoder还要awakeFromNib? awakeFromNib相较于initWithCoder的优势是:当awakeFromNib执行的时候,各种IBOutlet也都连接好 ...

  8. ajax+lazyload时lazyload失效问题及解决

    最近写公司的项目的时候遇到一个关于图片加载的问题,所做的页面是一个商城的商品列表页,其中需要显示商品图片,名称等信息,因为商品列表可能会很长,所以其中图片需要滑到可以显示的区域再进行加载. 首先我的图 ...

  9. Javascript Event事件中IE与标准DOM的区别

    1.事件流的区别 <body> <div> <button>点击这里</button> </div> </body> IE采用冒 ...

  10. CVE-2020-1947 Sharding-UI的反序列化复现及分析

    CVE-2020-1947 复现及分析 0x01 影响 Apache ShardingSphere < =4.0.0 0x02 环境搭建 incubator-shardingsphere 的ui ...