• bzoj2628: JZPSTR


    Description

    问题描述
      你要对一个字符串进行三种操作:
      0. 在位置x_i处插入一个字符串y_i
      1. 删除位置[x_i, y_i)的字符串
      2. 查询位置[x_i, y_i)的字符串包含多少次给定的子串z_i

    Input

      第一行,一个整数T,表示操作个数。
      下面T行,每行第一个数p_i,表示这个操作的类型:
      若p_i=0,则接下来有一个整数x_i和一个字符串y_i,表示进行插入操作;
      若p_i=1,则接下来有两个整数x_i和y_i,表示进行删除操作;
      若p_i=2,则接下来有两个整数x_i和y_i,以及一个字符串z_i,表示进行询问。
      字符串的下标从0开始(即第一个字符的下标为0)。
      初始时字符串为空。
      对于插入操作,插入后字符串y_i的首字符的下标应为x_i;
      对于删除操作,删除的区间[x_i, y_i)为左闭右开区间;
      对于查询操作,询问的区间[x_i, y_i)为左闭右开区间。
      所有插入的和查询的字符串均不为空,且只包含0~9的字符。
      所有询问的区间和删除的区间均不为空。
      保证输入数据合法。
      对于"左闭右开区间"不理解的可以去看样例解释。

    Output

      对每个询问操作,输出一行,表示这个询问的答案。

    暴力的复杂度约为1010,于是可以压位维护每个字符的所有出现位置,插入删除直接暴力,查询则可以通过移位和按位与实现。

    由于bitset封装的太好不便于高效区间操作所以还是手写一个。。

    #include<cstdio>
    #include<cstring>
    typedef unsigned long long u64;
    char buf[5000007],*ptr=buf-1;
    int n,o,x,y;
    char s[1000007];
    int c1[65555];
    const u64 N=1011111/64;
    u64 b[N];
    struct bitvec{
        u64 a[N];
        int len;
        void set(int x){a[x>>6]|=1llu<<x;}
        void reset(int x){a[x>>6]&=~(1llu<<x);}
        void set(int x,bool a){if(a)set(x);else reset(x);}
        bool test(int x){return a[x>>6]>>x&1;}
        void ins(int x,int y){
            len+=y;
            register int p=len+127>>6;
            int r=x+y+127>>6;
            int d1=y>>6,d2=y&63,d3=64-d2;
            if(d2)for(;p>=r;--p)a[p]=a[p-d1-1]>>d3|a[p-d1]<<d2;
            else for(;p>=r;--p)a[p]=a[p-d1];
            p=p+1<<6;
            while(p>x+y)--p,set(p,test(p-y));
        }
        void del(register int x,int y){
            len-=y;
            while(x<len&&(x&63))set(x,test(x+y)),++x;
            if(x==len)return;
            x>>=6;
            int r=len+127>>6;
            int d1=y>>6,d2=y&63,d3=64-d2;
            if(d2)for(;x<=r;++x)a[x]=a[x+d1]>>d2|a[x+d1+1]<<d3;
            else for(;x<=r;++x)a[x]=a[x+d1];
        }
        void cut(int x,int y){
            int r=y+127>>6,d1=x>>6,d2=x&63,d3=64-d2;
            if(d2)for(register int i=0;i<r;++i)b[i]&=a[d1+i]>>d2|a[d1+i+1]<<d3;
            else for(register int i=0;i<r;++i)b[i]&=a[d1+i];
        }
    }v[10];
    int _(){
        int x=0,c=*++ptr;
        while(c<48)c=*++ptr;
        while(c>47)x=x*10+c-48,c=*++ptr;
        return x;
    }
    void _(char*s){
        int c=*++ptr;
        while(c<48)c=*++ptr;
        while(c>47)*s++ =c,c=*++ptr;
        *s=0;
    }
    int main(){
        for(int i=1;i<65536;++i)c1[i]=c1[i>>1]+(i&1);
        fread(buf,1,sizeof(buf),stdin);
        n=_();
        while(n--){
            o=_();x=_();
            if(o==0){
                _(s);
                y=strlen(s);
                for(int i=0;i<10;++i){
                    v[i].ins(x,y);
                    for(int j=0;j<y;++j)v[i].set(x+j,s[j]=='0'+i);
                }
            }else if(o==1){
                y=_()-x;
                for(int i=0;i<10;++i)v[i].del(x,y);
            }else{
                y=_()-x;_(s);
                int l=strlen(s),ans=0;
                if(l<=y){
                    int d=y-l+1;
                    memset(b,-1,d+127>>3);
                    for(int i=0;i<l;++i)v[s[i]-'0'].cut(x+i,d);
                    while(d&63)--d,ans+=b[d>>6]>>d&1;
                    d>>=6;
                    for(int i=0;i<d;++i)ans+=c1[b[i]&65535]+c1[b[i]>>16&65535]+c1[b[i]>>32&65535]+c1[b[i]>>48];
                }
                printf("%d
    ",ans);
            }
        }
        return 0;
    }
  • 相关阅读:
    第三章第四章总结
    java学习2打架代码编写
    windows server 2008 远程桌面(授权、普通用户登录)
    Windows组建网络服务 ——WEB服务器的组建与架构
    windows server 2008 站点系列
    将 Ubuntu 加入到 Windows 2003 AD域
    Windows Server 2008组策略管理与配置
    AD用户设置系列
    利用windows 2003实现服务器群集的搭建与架设
    server2008 跨进新的平台(三)高端的备份还原工具
  • 原文地址:https://www.cnblogs.com/ccz181078/p/6289384.html
Copyright © 2020-2023  润新知