后缀自动机感觉好万能
tries图和ac自动机能做的,后缀自动机很多也都可以做
这里的循环匹配则是后缀自动机能做的另一个神奇功能
循环匹配意思就是S是abba, T是abb
问'abb', 'bba','bab'在S中出现过多少次。
我们先把T的末尾循环加一遍,变成abbab
然后把问题转换成,求T的每个后缀和S的最长公共子串
如果最长公共子串的长度大于等于T的长度,那么就说明这个后缀匹配成功
做法就是先对S建立一个后缀自动机,然后记录一个状态
(u, l),u表示当前在后缀自动机匹配的位置,l表示最长公共子串的长度
考虑转移的话,就是
如果下一个位置可以匹配,那么u就到相应的位置,l = l+1
答案更新的时候要注意,如果l大于T的长度len,就需要顺着link往前走到第一个能匹配的位置,即第一个maxlen[x] >= len的地方,然后答案加上endpos[x],不然会丢一部分答案。
如果下一个位置不可以匹配,那么u就顺着link边走,走到第一个能匹配的地方,如果找不到,那u就设成起点,l为0
还有一个问题就是串重复的情况,比如说T是aa,那么扩充就会变成aaa,aa和aa重复。
如果串重复的话,那么必定会到同一个状态,所以一个状态标记一下,只更新一遍答案就可以了
#include #include #include #include #include