zoukankan      html  css  js  c++  java
  • BZOJ 2555: SubString 后缀自动机 + LCT

    很水的一道题,就是有些细节没注意到.

    比如说将调试信息误以为是最终结果而多调了20分钟QAQ .....

    我们注意到,每新加一个节点,改变的是该节点沿着 Parent 走一直走到根节点.

    对应的,在 LCT 上进行修改即可.

    改变一个节点的 Parent,就对应 cut

    断掉原边后将新边连接即可.

    Code:

    #include <cstdio>
    #include <algorithm>
    #include <cstring>
    #define setIO(s) freopen(s".in","r",stdin) 
    #define maxn 2000009
    #define N 30 
    using namespace std;
    int result,query,n,M; 
    char s[3000009]; 
    
    void Decode(int mask){
        int nn=strlen(s); 
        for(int j=0;j<nn;j++){
            mask=(mask*131+j)%nn;    
            swap(s[j],s[mask]);
        }
    } 
    struct Link_Cut_Tree{
        int ch[maxn][2],f[maxn],tag[maxn],sta[maxn],val[maxn];
        int get(int x) {return ch[f[x]][1]==x; }
        int which(int x){ return ch[f[x]][1]==x;}
        int isRoot(int x){ return !(ch[f[x]][1]==x||ch[f[x]][0]==x);}
        int lson(int x){ return ch[x][0];}
        int rson(int x){return ch[x][1];}
        void add(int x,int delta){if(!x)return;val[x]+=delta,tag[x]+=delta;}
        void pushdown(int x){if(tag[x]) add(lson(x),tag[x]),add(rson(x),tag[x]),tag[x]=0;}
        void rotate(int x){
            int old=f[x],fold=f[old],which=get(x);
            if(!isRoot(old)) ch[fold][ch[fold][1]==old]=x;
            ch[old][which]=ch[x][which^1],f[ch[old][which]]=old;
            ch[x][which^1]=old,f[old]=x,f[x]=fold;
        }
        void splay(int x){
            int v=0,u=x;
            sta[++v]=u;
            while(!isRoot(u)) sta[++v]=f[u],u=f[u];
            while(v) pushdown(sta[v--]);
            u=f[u];
            for(int fa;(fa=f[x])!=u;rotate(x))
                if(f[fa]!=u) rotate(get(fa)==get(x)?fa:x);
        }
        void Access(int x){
            for(int y=0;x;y=x,x=f[x])
                splay(x),ch[x][1]=y;
        }
        void link(int a,int b){ 
            Access(a),splay(a),add(a,val[b]),f[b]=a;
        }
        void cut(int b){
            Access(b),splay(b);
            add(lson(b),-val[b]),f[lson(b)]=ch[b][0]=0; 
        }
    }tree;
    struct Suffix_Automaton{
        int last,tot,dis[maxn],ch[maxn][N];
        int f[maxn];
        void init(){last=tot=1;}
        void ins(int c){
            int p=last,np=++tot; last=np; dis[np]=dis[p]+1;tree.val[np]=tree.tag[np]=1;
            while(p&&!ch[p][c])ch[p][c]=np,p=f[p];
            if(!p) f[np]=1,tree.link(1,np); 
            else{
                int q=ch[p][c],nq; 
                if(dis[q]==dis[p]+1)f[np]=q,tree.link(q,np);
                else{
                    nq=++tot;tree.val[nq]=0;
                    dis[nq]=dis[p]+1;
                    memcpy(ch[nq],ch[q],sizeof(ch[q]));
                    //printf("%d
    ",q)
                    f[nq]=f[q];
                    tree.link(f[q],nq),tree.cut(q),tree.link(nq,q),tree.link(nq,np); 
                    f[q]=f[np]=nq;
                    while(p&&ch[p][c]==q) ch[p][c]=nq,p=f[p];
                }
            }
        }
        int search_str(){
            n=strlen(s);
            int p = 1;
            for(int i=0;i<n;++i) {
                if(!ch[p][s[i]-'A']) return -1;
                p=ch[p][s[i]-'A']; 
            }
            return p;
        }
    }sam;
    int main(){
        //setIO("input");
        sam.init(),scanf("%d%s",&query,s),n=strlen(s); 
        for(int i=0;i<n;++i) sam.ins(s[i]-'A');
        char opt[10]; int u; 
        while(query--){
            scanf("%s%s",opt,s);
            Decode(M); 
            if(opt[0]=='A') {
                n=strlen(s);
                for(int i=0;i<n;++i) sam.ins(s[i]-'A');
            }
            if(opt[0]=='Q') {
                u=sam.search_str();
                if(u==-1) result=0;
                else tree.Access(u),tree.splay(u),result=tree.val[u];
                M^=result;
                printf("%d
    ",result); 
            }
        }
        return 0;
    }
    

      

  • 相关阅读:
    PHPCMS V9 二次开发 —— 入口程序+系统类库与函数库调用+配置文件调用+函数扩展+控制器扩展技巧
    PHPCMS V9 二次开发 —— 二次开发流程
    PHPCMS V9 二次开发 —— 目录结构分析
    如何让pc端网站在手机上可以等比缩放的整个显示
    上海科技馆+博物馆两日游
    旅行的意义
    这样预判牛市第一波能涨到多少点比较靠谱
    历次牛市规律告诉你:如果只有五万元,买什么股票最赚钱?捕捉十年十倍大牛股
    算法竞赛在线测评 competitive programming
    不要做孩子学习的拐杖
  • 原文地址:https://www.cnblogs.com/guangheli/p/10341195.html
Copyright © 2011-2022 走看看