BZOJ 3172 [Tjoi2013]单词 AC自动机(fail树)

原创 2015年07月08日 14:28:36

题意:链接

方法: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\n",end[k]);
    }
}
版权声明:本文为博主原创文章,未经博主允许不得转载。

【BZOJ3172】【Tjoi2013】单词 AC自动机模板题

题解:水爆了,直接AC自动机瞎写就行。 坑:……时隔一个半月的感动AC,竟然是因为这道题可以有重复单词233。 代码: #include #include #include #include...
  • Vmurder
  • Vmurder
  • 2015年01月14日 16:16
  • 1676

【BZOJ】【P3172】【Tjoi2013】【单词】【题解】【AC自动机】

传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3172 指针模板 Code: #include #define idx(c) (c-'a...

【bzoj 3172】【codevs 2542】[Tjoi2013]单词(AC自动机)

泼墨绘花,把盏对茶,半生浮华,几番清雅?

后缀数组/AC自动机——BZOJ3172 [Tjoi2013]单词

题面:BZOJ3172 首先把论文给造出来是吧。。。(单词中间加空格好了) 然后在这篇文章(其实是一串字符串)求Height(后缀数组实现) 把Height数组求出来之后我们对于每个单词暴力向左...

BZOJ 3172 [Tjoi2013] 单词 [AC自动机模板]

AC自动机[模板]

BZOJ 3172 TJOI 2013 单词 AC自动机

题目大意:给出一个由几个单词组成的文章,问每一个单词在文章中出现过多少次。 思路:应该是后缀数组把,但是我还不会,先用AC自动机水过去,但是发现一点都不水。。几乎调了一下午。 首先是用所有...

BZOJ 3172 Tjoi2013 单词 fail树

题目大意及后缀数组做法见 http://blog.csdn.net/popoqqq/article/details/41042473 原来正解是fail树……难怪后缀数组被卡成这样 首先我们将给出...
  • PoPoQQQ
  • PoPoQQQ
  • 2014年12月08日 20:18
  • 1842

bzoj3172 ac自动机fail树应用

记录每一个点建自动机时候的访问次数。 建Fail树,然后节点子树的大小即为当前点出现的次数。#include #include #include #include #include using ...

bzoj3172.单词(AC自动机)

有 N 个单词组成了一篇文章,求每个单词在这篇文章中出现了多少次。 多个字符串匹配的问题,建立AC自动机。如果某个单词在 i 节点出现了,那么在 i 节点 fail 指针所指节点也出现过。 ...
  • ALXPCUN
  • ALXPCUN
  • 2016年04月22日 14:11
  • 133

【BZOJ3172】单词,AC自动机练习

用沉默对付他人的非议,在强者是一种风度,在弱者是一种聪明。
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:BZOJ 3172 [Tjoi2013]单词 AC自动机(fail树)
举报原因:
原因补充:

(最多只允许输入30个字)