其他分享
首页 > 其他分享> > BZOJ4231 回忆树

BZOJ4231 回忆树

作者:互联网

用 \(kmp\) 暴力处理经过 \(lca\) 的匹配,这一部分复杂度为 \(O(\sum|s|)\)。然后就只用考虑直上直下的链的匹配,离线后对询问串建 \(AC\) 自动机,在原树上遍历时加入贡献,答案差分统计,即长链的匹配减去短链的匹配,用树状数组维护 \(fail\) 树子树和即可。

#include<bits/stdc++.h>
#define maxn 300010
#define lowbit(x) (x&(-x))
using namespace std;
template<typename T> inline void read(T &x)
{
    x=0;char c=getchar();bool flag=false;
    while(!isdigit(c)){if(c=='-')flag=true;c=getchar();}
    while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
    if(flag)x=-x;
}
int n,m,root;
int ans[maxn],f[maxn][19],dep[maxn],nxt[maxn];
char s[maxn],col[maxn],str[maxn],tmp[maxn];
struct node
{
    int p,v,id;
    node(int a=0,int b=0,int c=0)
    {
        p=a,v=b,id=c;
    }
};
vector<node> q1[maxn],q2[maxn];
struct edge
{
    int to,nxt;
    char v;
    edge(int a=0,int b=0,char c=0)
    {
        to=a,nxt=b,v=c;
    }
}e[maxn];
int head[maxn],edge_cnt;
void add(int from,int to,char val)
{
    e[++edge_cnt]=edge(to,head[from],val),head[from]=edge_cnt;
}
struct AC
{
    int tot,cnt;
    int ch[maxn][28],fail[maxn],in[maxn],out[maxn],tr[maxn];
    vector<int> ve[maxn];
    void update(int x,int v)
    {
        if(!x) return;
        x=in[x];
        while(x<=n) tr[x]+=v,x+=lowbit(x);
    }
    int ask(int x)
    {
        int v=0;
        while(x) v+=tr[x],x-=lowbit(x);
        return v;
    }
    int query(int x)
    {
        return ask(out[x])-ask(in[x]-1);
    }
    int insert(int type=0)
    {
        int p=root,len=strlen(s+1);
        if(type) reverse(s+1,s+len+1);
        for(int i=1;i<=len;++i)
        {
            int c=s[i]-'a';
            if(!ch[p][c]) ch[p][c]=++tot;
            p=ch[p][c];
        }
        if(type) reverse(s+1,s+len+1);
        return p;
    }
    void dfs_dfn(int x)
    {
        in[x]=++cnt;
        for(int i=0;i<ve[x].size();++i) dfs_dfn(ve[x][i]);
        out[x]=cnt;
    }
    void build()
    {
        queue<int> q;
        for(int c=0;c<26;++c)
            if(ch[root][c])
                q.push(ch[root][c]);
        while(!q.empty())
        {
            int x=q.front();
            q.pop();
            for(int c=0;c<26;++c)
            {
                int y=ch[x][c];
                if(y) fail[y]=ch[fail[x]][c],q.push(y);
                else ch[x][c]=ch[fail[x]][c];
            }
        }
        for(int i=1;i<=tot;++i) ve[fail[i]].push_back(i);
        dfs_dfn(root);
    }
}A,B;
void dfs_pre(int x,int fa)
{
    dep[x]=dep[f[x][0]=fa]+1;
    for(int i=1;i<=17;++i) f[x][i]=f[f[x][i-1]][i-1];
    for(int i=head[x];i;i=e[i].nxt)
    {
        int y=e[i].to;
        if(y==fa) continue;
        col[y]=e[i].v,dfs_pre(y,x);
    }
}
int lca(int x,int y)
{
    if(dep[x]<dep[y]) swap(x,y);
    for(int i=17;i>=0;--i)
        if(f[x][i]&&dep[f[x][i]]>=dep[y])
            x=f[x][i];
    if(x==y) return x;
    for(int i=17;i>=0;--i)
        if(f[x][i]&&f[x][i]!=f[y][i])
            x=f[x][i],y=f[y][i];
    return f[x][0];
}
int get(int x,int k)
{
    for(int i=0;i<=17;++i)
        if((k>>i)&1)
            x=f[x][i];
    return x;
}
void work(int x,int y,int id)
{
    int anc=lca(x,y),p1=A.insert(),p2=B.insert(1),len=strlen(s+1),p,cnt1=0,cnt2=0,pos=0;
    p=get(x,max(dep[x]-dep[anc]-len+1,0));
    q2[x].push_back(node(p2,1,id)),q2[p].push_back(node(p2,-1,id));
    while(p!=anc) str[++cnt1]=col[p],p=f[p][0];
    p=get(y,max(dep[y]-dep[anc]-len+1,0));
    q1[y].push_back(node(p1,1,id)),q1[p].push_back(node(p1,-1,id));
    while(p!=anc) tmp[++cnt2]=col[p],p=f[p][0];
    for(int i=cnt2;i;--i) str[++cnt1]=tmp[i];
    for(int i=1;i<=len;++i) nxt[i]=0;
    for(int i=2;i<=len;++i)
    {
        while(pos&&s[pos+1]!=s[i]) pos=nxt[pos];
        nxt[i]=(pos+=s[pos+1]==s[i]);
    }
    pos=0;
    for(int i=1;i<=cnt1;++i)
    {
        while(pos&&s[pos+1]!=str[i]) pos=nxt[pos];
        pos+=s[pos+1]==str[i];
        if(pos==len) ans[id]++,pos=nxt[pos];
    }
}
void dfs_ans(int x,int p1,int p2)
{
    A.update(p1,1),B.update(p2,1);
    for(int i=0;i<q1[x].size();++i)
        ans[q1[x][i].id]+=A.query(q1[x][i].p)*q1[x][i].v;
    for(int i=0;i<q2[x].size();++i)
        ans[q2[x][i].id]+=B.query(q2[x][i].p)*q2[x][i].v;
    for(int i=head[x];i;i=e[i].nxt)
    {
        int y=e[i].to,v=e[i].v-'a';
        if(y==f[x][0]) continue;
        dfs_ans(y,A.ch[p1][v],B.ch[p2][v]);
    }
    A.update(p1,-1),B.update(p2,-1);
}
int main()
{
    read(n),read(m);
    for(int i=1;i<n;++i)
    {
        int x,y;
        read(x),read(y),scanf("%s",s),add(x,y,s[0]),add(y,x,s[0]);
    }
    dfs_pre(1,0);
    for(int i=1;i<=m;++i)
    {
        int x,y;
        read(x),read(y),scanf("%s",s+1);
        if(x!=y) work(x,y,i);
    }
    A.build(),B.build(),dfs_ans(1,root,root);
    for(int i=1;i<=m;++i) printf("%d\n",ans[i]);
    return 0;
}

标签:anc,int,dep,while,edge,maxn,BZOJ4231,回忆
来源: https://www.cnblogs.com/lhm-/p/14220945.html