码迷,mamicode.com
首页 > 其他好文 > 详细

luogu_3966【题解】单词 AC自动机

时间:2019-07-06 20:57:33      阅读:150      评论:0      收藏:0      [点我收藏+]

标签:turn   show   ace   printf   表示   c++   自动机   space   pre   

题面: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自动机

标签:turn   show   ace   printf   表示   c++   自动机   space   pre   

原文地址:https://www.cnblogs.com/ChrisKKK/p/11144072.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!