zoukankan      html  css  js  c++  java
  • BZOJ 3172 [Tjoi2013]单词 AC自己主动机(fail树)

    题意:链接

    方法:AC自己主动机与fail树性质

    解析:复习AC自己主动机的第一道题?(真正的第一题明明是又一次写了遍hdu2222!

    )

    这题说实话第一眼看上去就是个sb题,仅仅要建出来自己主动机。然后搜fail树即可了。只是看完140142的博客貌似这样会T?只是他也过了是什么鬼?反正想想后没想到什么好的方法就去看了看题解。写题解的大牛们的思路能够概括成一句话,也就是fail树的性质:

    你要查找某个串的出现次数则为该串的根节点在fail树上出现的次数之和。

    恩知道了这个性质后(能够yy下),这道题能够再来个优化,也就是在build的时候能够将这个整个树的bfs序求出来,又由于fail节点都是后面的连到前面的,所以无后效性,也就是说,我们能够从这个栈的栈顶抽元素,并将他的end值加到他的fail节点上。这样也许会快非常多?

    输出部分就是我们之前说的了。找到每一个串的根节点。询问他的end值即可。

    代码:

    
    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    #define N 210
    #define M 1000100
    using namespace std;
    int n,L,root,size;
    char s[N][M];
    int next[M][27],fail[M],end[M],q[M];
    int newnode()
    {
        for(int i=1;i<=26;i++)next[size][i]=-1;
        end[size++]=0;
        return size-1;
    }
    void init()
    {
        size=0;
        root=newnode();
    }
    void ins()
    {
        int l=strlen(s[L]);
        int now=root;
        for(int i=0;i<l;i++)
        {
            int tmp=s[L][i]-'a'+1;
            if(next[now][tmp]==-1)next[now][tmp]=newnode();
            now=next[now][tmp];
            end[now]++;
        }
    }
    void build()
    {
        int LL=0,RR=-1;
        for(int i=1;i<=26;i++)
        {
            if(next[root][i]==-1)next[root][i]=root;
            else
            {
                fail[next[root][i]]=root;
                q[++RR]=next[root][i];
            }
        }
        while(LL<=RR)
        {
            int u=q[LL++];
            for(int i=1;i<=26;i++)
            {
                if(next[u][i]==-1)next[u][i]=next[fail[u]][i];
                else
                {
                    fail[next[u][i]]=next[fail[u]][i];
                    q[++RR]=next[u][i];
                }
            }
        }
        for(int i=RR;i>=0;i--)
        {
            end[fail[q[i]]]+=end[q[i]];
        }
    }
    int main()
    {
        init();
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
        {
            scanf("%s",s[i]);
            L++;
            ins();
        }
        build();
        L=1;
        for(int i=1;i<=n;i++)
        {
            int k=root;
            int l=strlen(s[i]);
            for(int j=0;j<l;j++)k=next[k][s[i][j]-'a'+1];
            printf("%d
    ",end[k]);
        }
    }
  • 相关阅读:
    CompareUtil
    linux awk学习笔记
    linux用grep查找包含两个关键字的命令
    mysql 使用set names 解决乱码问题
    对私有静态方法进行单测
    使用JUnit测试预期异常
    Tortoise svn 冲突解决主要办法
    tortoise svn冲突解决
    word-break与word-wrap
    移动端适配
  • 原文地址:https://www.cnblogs.com/blfshiye/p/5418418.html
Copyright © 2011-2022 走看看