zoukankan      html  css  js  c++  java
  • 【BZOJ-3172】单词 AC自动机

    3172: [Tjoi2013]单词

    Time Limit: 10 Sec  Memory Limit: 512 MB
    Submit: 2567  Solved: 1200
    [Submit][Status][Discuss]

    Description

    某人读论文,一篇论文是由许多单词组成。但他发现一个单词会在论文中出现很多次,现在想知道每个单词分别在论文中出现多少次。

    Input

    第一个一个整数N,表示有多少个单词,接下来N行每行一个单词。每个单词由小写字母组成,N<=200,单词长度不超过10^6

    Output

    输出N个整数,第i行的数字表示第i个单词在文章中出现了多少次。

    Sample Input

    3
    a
    aa
    aaa

    Sample Output

    6
    3
    1

    HINT

    Source

    Solution

    ACMachine模板题,而且不用匹配.

    建Trie树,构建Fail指针,在构建Fail指针的过程中,累加一下次数即可

    Code

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cmath>
    #include<cstring>
    using namespace std;
    #define N 1000010
    int sz,son[N][26],fail[N],q[N],sum[N],loc[N];
    int n; char s[N];
    void clear()
    {
        sz=1;
        for (int i=0; i<26; i++) son[0][i]=1;
    }
    void insert(int &pos)
    {
        int x=1;
        for (int i=0; i<strlen(s); i++)
            {
                if (!son[x][s[i]-'a']) son[x][s[i]-'a']=++sz;
                x=son[x][s[i]-'a']; sum[x]++;
            }
        pos=x;
    }
    void buildfail()
    {
        int head=0,tail=1; q[0]=1; fail[1]=0;
        while (head<tail)
            {
                int now=q[head++];
                for (int i=0; i<26; i++)
                    {
                        if (!son[now][i]) continue;
                        int ff=fail[now];
                        while(!son[ff][i]) ff=fail[ff];
                        fail[son[now][i]]=son[ff][i];
                        q[tail++]=son[now][i];
                    }
            }
         for (int i=tail-1; i>=0; i--)
             sum[fail[q[i]]]+=sum[q[i]];
    }
    int main()
    {
        clear();
        scanf("%d",&n); 
        for (int i=1; i<=n; i++)
            scanf("%s",s),insert(loc[i]);
        buildfail();
        for (int i=1; i<=n; i++)
            printf("%d
    ",sum[loc[i]]);
        return 0;
    }

    很早以前做的题了,那时候用自动机,出现奇怪的错误...interesting...MDZZ

  • 相关阅读:
    [计算机网络-传输层] 无连接传输:UDP
    [BinaryTree] 最大堆的类实现
    [OS] 生产者-消费者问题(有限缓冲问题)
    [剑指Offer] 64.滑动窗口的最大值
    [剑指Offer] 63.数据流中的中位数
    [剑指Offer] 62.二叉搜索树的第k个结点
    [OS] CPU调度
    [剑指Offer] 60.把二叉树打印成多行
    MySQL数据库实验二:单表查询
    数据库实验:基本表的定义与修改
  • 原文地址:https://www.cnblogs.com/DaD3zZ-Beyonder/p/5350135.html
Copyright © 2011-2022 走看看