首页 > 其他分享 >AC 自动机

AC 自动机

时间:2023-02-02 17:47:49浏览次数:33  
标签:状态 AC now trie fail 自动机

概述

  • 自动机是一种接受一个有序序列并对之进行判定的数学模型。

  • 一般来讲,自动机可以抽象为一个有向图,点为状态,边为某种转移。

  • 许多字符串算法的本质和关键,都是利用自动机的思想来探寻不同条件下的不同转移,以设法利用已求出的状态。

确定有限状态自动机(DFA)

  • 1.组成:

    • (1) 字符集 \(\Sigma\),该自动机只能输入这些字符。

    • (2) 状态集 \(Q\)。有向图意义下,DFA 中的状态就相当于图上的顶点。

    • (3) 起始状态 \(start\), \(start\in Q\)。\(start\) 是一个特殊的状态。起始状态一般用 \(s\) 表示,为了避免混淆(字符串是 \(s\) ),本文中使用 \(start\)。

    • (4) 接受状态集 \(F\),\(F\subseteq Q\) ,是一组特殊的状态。该 DFA 仅接受这些状态。

    • (5) 转移函数 \(\delta(v,c)=v'\),\(\delta\) 是一个接受两个参数返回一个值的函数,其中第一个参数和返回值都是一个状态,第二个参数是字符集中的一个字符。有向图意义下, DFA 中的转移函数就相当于顶点间的边,而每条边上都有一个字符。

      • 当读到的字符不满足当前状态 \(v\) 的任意一个转移函数 \(\delta(v,c)\) ,可以认为有一条虚边 \(\delta(v,c)=null\) , 且 \(\delta(null,all)=null,null\notin F\) 。

      • 可以定义扩展转移函数 \(\delta(v,s)=v'\) ,其第二个参数为字符串。

      • 实际应用中,常常会有自环等操作。

  • 2.作用:判断一个字符串是否满足某些条件(即字符串结束时处在一个接受状态)。可以看出,DFA 是在线的。

  • 3.代表:trie;KMP;AC 自动机。

AC 自动机

概述

  • 坐,都坐。我知道各位都很激动,但这个缩写的全称是 Aho-Corasick automaton,AC 是两位发明者的姓氏,automaton:自动机。

  • AC 自动机通过将 trie 的结构和 KMP 的思想结合起来,利用由 fail 指针重构的 trie 图,高效解决多个模式串相对文本串的各种出现问题。

  • 记 \(\Sigma\) 为字符集,则 AC 自动机的空间复杂度和建立时间复杂度均为 \(O(\sum |P|\times |\Sigma|)\)。这里认为预先输入的是模式串,询问相关的是文本串,因一般是求输入的串在询问串中的出现。

实现原理

  • trie 部分参见“字典树”。

  • 朴素实现:

    • \(fail\) 数组表示当前节点代表的状态的最大有效后缀对应的节点。

      • 我们需要谈谈什么是有效。考虑一个只插入了 \(abca\) 的 AC 自动机:

      • \(abca\) 的最大真后缀显然是 \(bca\),\(bca\) 的是 \(ca\),\(ca\) 的是 \(a\)。但 \(bca,ca\) 对我们来说都是无效的:它不是任何一个模式串的前缀

      • 事实上,我们不妨把 \(bca,ca\) 都建出来:我们会发现,这里 \(abca\) 是“关键点”,而我们只要保留“关键点”相关的点——这是个虚树!而这里的相关,或者说“lca”,是前缀。

      • 这也是 AC 自动机必须全部插完之后才构建的理由:如果在线,那么旧有的 fail 边可能是错误的,并没有指向最大有效后缀。

    • 当查询某个文本串时,如果发现不存在对应的 \(e_{now,c}\),那么 \(now=fail_{now}\) 并再次尝试,直到匹配成功或 \(now=0\) 且仍然匹配失败,此时放弃前面所有的字符从头开始。

    • 实际形态的话,我们看一张图。黄点为字符串结束点,黑边为 trie 边,红边为 fail 边。

    • 称黑边即 trie 边构成的树为 trie 树,红边即 fail 边构成的树为 fail 树。两者是 AC 自动机的灵魂;trie 图也只是由这两者共同递推出的罢了。

  • trie 图实现:

    • 容易看出,上面这个算法的主要弊病在于可能会连续多次失配,于是回跳次数太多。

    • 考虑路径压缩!

    • 显而易见 \(dep_{fail_{now}}<dep_{now}\),从而我们使用 bfs 实现,并设法让子节点的 \(fail\) 利用上父节点的 \(fail\)。

    • 我们站在父亲处理儿子。假如我有这个儿子,那么 \(fail_{e_{now,c}}=e_{fail_{now},c}\),即我儿子的最大有效后缀 \(=\) 我的最大有效后缀 \(+c\)。

    • 咦?万一我的最大有效后缀没有对应 \(c\) 的边怎么办?

    • 这就不得不谈及非常妙的第二种操作了。假如我没有这个儿子,那么 \(e_{now,c}=e_{fail_{now},c}\)!让虚边来承担 \(fail\),或者说,把 \(fail\) 的效果等效到 \(e\) 里。

    • 考虑 \(fail_{now}\) 没有对应 \(c\) 的转移的情况,此时我们有 \(e_{fail_{now},c}=e_{fail_{fail_{now}},c}\)!

    • 即,这实质上不是 \(fail\) 了一次,而是不断 \(fail\) 直到找到匹配。

    • 如果一直找不到匹配,则最终会达到根节点的某个子节点,从而有 \(e_{fail_{now},c}=e_{0,c}=0=rt\),抛弃了所有前缀。

    • 相当于这是一种自动的路径压缩。相较于并查集的由儿子递归地压缩父亲,这里父亲已经压好了,儿子只要连上去就相当于把自己也压了(神乎其技!)。

    • 为了避免把每个 \(dep=1\) 的节点的 \(fail\) 以及无匹配的 \(e_{rt,c}\) 都设为 \(1\) 繁琐特判,我们选择令 \(rt=0\)。

struct AC_automaton{
	int e[maxn][26],tot;//边。动态开点所用。特别地,为了方便(指fail数组的建立),这里的rt=0 
	int cnt[maxn],fail[maxn];//当前点结尾的字符串数量。当前节点的最大有效后缀对应的节点。
	il void ins(string &s){
		int now=0,to=s.size()-1;
		For(i,0,to){
			if(!e[now][s[i]-'a'])
				e[now][s[i]-'a']=++tot;
			now=e[now][s[i]-'a'];
		}
		++cnt[now]; return;
	}
	int q[maxn],hd,tl;
	il void build(){
		hd=1,tl=0;
		For(i,0,25)
			if(e[0][i])
				q[++tl]=e[0][i];
		while(tl>=hd){
			int now=q[hd++];
			For(i,0,25)
				if(e[now][i]){
					fail[e[now][i]]=e[fail[now]][i];
					q[++tl]=e[now][i];
				}
				else e[now][i]=e[fail[now]][i];
		}
	}
};

应用与例题

  • \(\text{P5357 [模板]AC 自动机(二次加强版)}\)

    • 题意:求文本串 \(T\) 中 \(n\) 个模式串 \(P_{1\sim n}\) 分别的出现次数。

    • 数据范围:\(n\leqslant 2\times 10^5,\sum |P|\leqslant 2\times 10^5,|T|\leqslant 2\times 10^6\)。

    • 直觉上我们可以在 trie 图上,即按 \(e\),把 \(T\) 跑一遍。访问到的点标上 \(vis\)。但这不对。

      • 我们知道这里的 \(e_{now,c}\) 很多本质上都是 \(fail\) 边的等效,是舍弃了一部分前缀之后再接一个的结果。

      • 通过这些边走到的点,其实相当于从根走到了那个点,这条链上的所有点都是文本串的一部分!

    • 考虑一种暴力做法:每到一个点,暴力回跳 \(fail\),每个点都打 \(vis\)。

      • 啊...复杂度很不乐观啊,极限情况(虽然不可能)为 \(\sum |P|\times AC.dep\)。

      • 考虑把指向没有打 \(vis\) 必要的点(不是字符串结尾)的 \(fail\) 压缩成另一个 \(FAIL\) 用于回跳。

        • 假优化。这上界是一样的。
    • 铛铛!正解:tree dp!(???)

      • 考虑把 \(fail\) 数组对应的 \(fail\) 树建出来(不是 \(fail\) 边这种反边构成的内向树,是正边,正常的可以 dfs 的外向树)。

      • 而我们假做法中打的 \(vis\) 的含义其实是从这里到根的路径都 \(vis\) 了,就是说任意的直接/间接 fail...任意的有效后缀,即可能是某个模式串前缀的后缀,都 \(vis\) 了...咦?

        • tree dp,\(rvis_{now}=\sum rvis_{sons}+vis_{now}\)。

        • 从而我们有一个稳定的单次询问复杂度为 \(O(|T|+|AC|)\)。

  • \(\text{P2292 [HNOI2004] L 语言}\)

    • 题意:给出 \(n\) 个模式串 \(P_{1\sim n}\),给出 \(Q\) 组询问,每组询问形如求 \(T\) 的最大可理解前缀长度。所谓“可理解”,指的是可以划分成若干个可重的模式串。

    • 数据范围:\(n,|P|\leqslant 20,Q\leqslant 50,|T|\leqslant 2\times 10^6\)。实际上似乎并没有 \(10^8\) 的输入,不然不管怎么挣扎都 T 了。

    • 首先我们肯定不能上去跑,这就退化成暴搜了,多种并行决策没法决策。

    • 故考虑设计 dp:

      • 状态设计:\(dp_i\) 表示长为 \(i\) 的前缀是否可理解。

      • 初始化:\(dp_0=1\)。

      • 状态转移:...我们下面慢慢说。

    • 首先考虑一个朴素转移:\(dp_i=\operatorname{or}_ {j<i} dp_j\And [T_{j+1\sim i}\in P]\)。

    • 我们可以暴力枚举是 \(P\) 中那个,然后用字符串 hash 来比较。

    • 然而,端下去罢。\(\sum |T|\sim 10^8\),你怎么敢转移不 \(O(1)\) 的?你怎么敢?

    • 考虑利用 AC 自动机的性质。既然 \(|P|\leqslant 20\),就把 \(T_{i-19,i}\) 扔上去跑一遍,利用预处理的 \(rsta_{now}=rsta_{fail_{now}}|sta_{now}\) 可以 \(O(40)\) 检查出来每个串是否合法。

    • 不行,还是不够快。我们老是和“哪个 \(P\)”过不去,这是没前途的。设法把它优化掉。

    • 首先,上面那个。既然可以跑 \(T_{i-19,i}\),那岂不是也可以 \(O(1)\) 递推到 \(T_{i-18,i+1}\)?毕竟过程中经过的点其实是啥用都没有,我们关心的是它最后停在了哪里。

    • 它是个 \(20\) 位二进制状态。那么,我们把 \(dp_{i-20\sim i-1}\) 也压成 \(20\) 位二进制状态!显然它也可以 \(O(1)\) 递推。两者 \(\And\) 的结果就是 \(dp_i\)。

    • 好了,解决了。\(O(\sum |T|)\)。

标签:状态,AC,now,trie,fail,自动机
From: https://www.cnblogs.com/weixin2024/p/17086792.html

相关文章

  • 【2023.02.02】PVE安装MacOS后的设置
    系统内设置安装好系统后进入终端输入diskutillist可以得到现在的硬盘信息disk0是安装系统的虚拟硬盘disk0s1是该硬盘第一个分区disk2是Opencore镜像盘disk2s1是......
  • 5G's impact on business
    By2024,morethan40%oftheworld'spopulationwillbeusing5Gtechnology.Theideaofsuchwidespreadadoptionisfascinating,buttherolloutof5Ghaspr......
  • package.json 的配置文件
                         ......
  • greenplum数据库vacuum操作
    一:vacuum什么是vacuumvacuum是greenplum数据库中用来回收死亡元组占用空间的语句。 标准语句:VACUUM[FULL][FREEZE][VERBOSE][table] VACUUM[FULL][FREEZE......
  • mac编译wat报错解决方案
    按照github的步骤一步步来的时候,最后一步出现问题,ld:cannotlinkdirectlywithdylib/framework,yourbinaryisnotanallowedclientof/usr/lib/libcrypto.dylib......
  • ElasticSearch、kibana、logstach部署
    ElasticSearch+NLog实现https://blog.csdn.net/weixin_51439775/article/details/128539623https://www.cnblogs.com/piscesLoveCc/p/7230426.htmlElasticSearch、kibana......
  • 常用的anaconda命令记录
    conda命令condacreate--nametestpython=3.7建立一个名字叫做test的,python版本为3.7的新环境。condaenvlist查看conda中的所有已安装的环境。pip命令pipins......
  • nginx 日志分析之 access.log 格式详解
    说明:access.log的格式是可以自己自定义,输出的信息格式在nginx.conf中设置一般默认配置如下:http{...log_formatmain'$remote_addr-$remote_user[$time_lo......
  • 【2023.02.01】在PVE上安装MacOS 13 Ventura
    【2023.02.01】在PVE上安装MacOS13Ventura本文参考链接:InstallingmacOS13VenturaonProxmox7.2–NicholasSherlock本次平台是i99980hk,CPU尽量新一点应该都可......
  • 【笔记向】package.json main 作用
    package.jsonmain作用在package.json文件中,"main"字段指定了这个包在被其他包依赖时,入口文件的文件名。例如,如果在package.json中的"main"字段被设置为"index.......