[BZOJ2780][Spoj8093]Sevenk Love Oimaster(广义后缀自动机)

题目描述

传送门

题解

对于所有的模板串建立广义后缀自动机,对于每一个点统计一下right集合中有几个模板串
然后对于所有的查询串在后缀自动机上匹配,然后看一下最终匹配到的那个点有几个模板串就行了
具体的方法是记录一下每一个点最后匹配到的是哪个模板串,然后记录一个cnt,每一次暴力顺着pre指针往上跳,跳到已经更新过的就退出
时间复杂度不大好算,其实就是每一次加入一个串自动机上的点最多被遍历一次

感觉这题AC自动机也是很好做的,把所有的查询串扔到trie里建fail树,然后把模板串在自动机上匹配,每一次匹配到的所有点到根的路径上的所有查询串的end的答案都应该+1,这样区间修改单点查询;其实可以转化成单点修改然后转化为求子树权值和,并且因为每一次修改的点应该是一个并集,所以要按照dfs序排序然后容斥一下,不过是静态的最后只需要一次dfs就能出解了
本来还是想写一下的,但是这题先给的模板串再给的查询串,存起来比较麻烦,懒得写了

代码

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
using namespace std;
#define N 200005

char s[N];
int T,Q,n,root,sz,last,p,np,q,nq,ans;
int ch[N][130],pre[N],step[N],lc[N],cnt[N];

void extend(int id)
{
    last=root;
    for (int i=0;i<n;++i)
    {
        int x=s[i];
        p=last;np=++sz;last=np;
        step[np]=step[p]+1;
        while (p&&!ch[p][x])
        {
            ch[p][x]=np;
            p=pre[p];
        }
        if (!p) pre[np]=root;
        else
        {
            q=ch[p][x];
            if (step[q]==step[p]+1) pre[np]=q;
            else
            {
                nq=++sz;
                step[nq]=step[p]+1;
                memcpy(ch[nq],ch[q],sizeof(ch[q]));
                pre[nq]=pre[q];
                lc[nq]=lc[q];cnt[nq]=cnt[q];
                while (ch[p][x]==q)
                {
                    ch[p][x]=nq;
                    p=pre[p];
                }
                pre[np]=pre[q]=nq;
            }
        }
        p=np;
        while (p&&lc[p]!=id)
        {
            lc[p]=id;
            ++cnt[p];
            p=pre[p];
        }
    }
}
int sam()
{
    p=root;
    for (int i=0;i<n;++i)
    {
        int x=s[i];
        if (!ch[p][x]) return 0;
        else p=ch[p][x];
    }
    return cnt[p];
}
int main()
{
    scanf("%d%d",&T,&Q);
    root=++sz;
    while (T)
    {
        scanf("%s",s);n=strlen(s);
        extend(T);
        --T;
    }
    while (Q--)
    {
        scanf("%s",s);n=strlen(s);
        ans=sam();
        printf("%d\n",ans);
    }
}
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值