luogu_3966题解单词 AC自动机

Posted chriskkk

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了luogu_3966题解单词 AC自动机相关的知识,希望对你有一定的参考价值。

题面:https://www.luogu.org/problemnew/show/P3966

大意:小张最近在忙毕设,所以一直在读论文。一篇论文是由许多单词组成但小张发现一个单词会在论文中出现很多次,他想知道每个单词分别在论文中出现了多少次。

这次没有文本串。

用了个奇妙的方法。

a[x] 表示第x个串结尾的位置。

在bfs搜索求fail的时候手写,将所有的点都保存下来了。

num(x)则为经过这个点x的个数。

自然num(fail(x))+=num(x)。

所有经过x的串都能经过fail(x)。

最后求所有a[x]的num就是答案了。

代码如下。

#include<bits/stdc++.h>
#define sc(x) scanf("%d",&x)
using namespace std;
const int maxn=1e6+10;
int n,cnt;
struct trie
    int fail,num;
    int ch[30];
    #define fa(x) ac[x].fail
    #define nu(x) ac[x].num
ac[maxn];
int a[maxn],h[maxn];
inline void insert(string s,int x)
    int len=s.length();
    int now=0;
    for(int i=0;i<len;i++)
        if(!ac[now].ch[s[i]-a])
            ac[now].ch[s[i]-a]=++cnt;
        now=ac[now].ch[s[i]-a];
        nu(now)++;
    
    a[x]=now;

//bfs
inline void getfail()
    int head=0;int tail=0;
    for(int i=0;i<26;i++)
        if(ac[0].ch[i])
            fa(ac[0].ch[i])=0,h[++tail]=ac[0].ch[i];
    while(head<tail)
        int now=h[++head];
        for(int i=0;i<26;i++)
            if(ac[now].ch[i])
                fa(ac[now].ch[i])=ac[fa(now)].ch[i];
                h[++tail]=ac[now].ch[i];
            
            else
                ac[now].ch[i]=ac[fa(now)].ch[i];
        
    
 
inline void ask()
    for(int i=cnt;i>=0;i--) nu(fa(h[i]))+=nu(h[i]);
    for(int i=1;i<=n;i++) printf("%d\n",nu(a[i]));

int main()

    sc(n);
    string s;
    for(int i=1;i<=n;i++)
        cin>>s;
        insert(s,i);
    
    fa(0)=0;
    getfail();
    ask();
    // while(1);
    return 0;

话说和网上大佬学的struct 存trie感觉挺有意思。

改不过来了。

以上是关于luogu_3966题解单词 AC自动机的主要内容,如果未能解决你的问题,请参考以下文章

P3966 [TJOI2013]单词(AC自动机,fail树)

P3966 [TJOI2013]单词(AC自动机,fail树)

题解单词

P3966 [TJOI2013]单词

题解Luogu P4052[JSOI2007]文本生成器 AC自动机

luogu P3808 模板AC自动机(简单版)