Description
问题描述
你要对一个字符串进行三种操作:
0. 在位置x_i处插入一个字符串y_i
1. 删除位置[x_i, y_i)的字符串
2. 查询位置[x_i, y_i)的字符串包含多少次给定的子串z_i
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的字符。
所有询问的区间和删除的区间均不为空。
保证输入数据合法。
对于"左闭右开区间"不理解的可以去看样例解释。
下面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\n",ans); } } return 0; }