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[],*ptr=buf-;
int n,o,x,y;
char s[];
int c1[];
const u64 N=/;
u64 b[N];
struct bitvec{
u64 a[N];
int len;
void set(int x){a[x>>]|=1llu<<x;}
void reset(int x){a[x>>]&=~(1llu<<x);}
void set(int x,bool a){if(a)set(x);else reset(x);}
bool test(int x){return a[x>>]>>x&;}
void ins(int x,int y){
len+=y;
register int p=len+>>;
int r=x+y+>>;
int d1=y>>,d2=y&,d3=-d2;
if(d2)for(;p>=r;--p)a[p]=a[p-d1-]>>d3|a[p-d1]<<d2;
else for(;p>=r;--p)a[p]=a[p-d1];
p=p+<<;
while(p>x+y)--p,set(p,test(p-y));
}
void del(register int x,int y){
len-=y;
while(x<len&&(x&))set(x,test(x+y)),++x;
if(x==len)return;
x>>=;
int r=len+>>;
int d1=y>>,d2=y&,d3=-d2;
if(d2)for(;x<=r;++x)a[x]=a[x+d1]>>d2|a[x+d1+]<<d3;
else for(;x<=r;++x)a[x]=a[x+d1];
}
void cut(int x,int y){
int r=y+>>,d1=x>>,d2=x&,d3=-d2;
if(d2)for(register int i=;i<r;++i)b[i]&=a[d1+i]>>d2|a[d1+i+]<<d3;
else for(register int i=;i<r;++i)b[i]&=a[d1+i];
}
}v[];
int _(){
int x=,c=*++ptr;
while(c<)c=*++ptr;
while(c>)x=x*+c-,c=*++ptr;
return x;
}
void _(char*s){
int c=*++ptr;
while(c<)c=*++ptr;
while(c>)*s++ =c,c=*++ptr;
*s=;
}
int main(){
for(int i=;i<;++i)c1[i]=c1[i>>]+(i&);
fread(buf,,sizeof(buf),stdin);
n=_();
while(n--){
o=_();x=_();
if(o==){
_(s);
y=strlen(s);
for(int i=;i<;++i){
v[i].ins(x,y);
for(int j=;j<y;++j)v[i].set(x+j,s[j]==''+i);
}
}else if(o==){
y=_()-x;
for(int i=;i<;++i)v[i].del(x,y);
}else{
y=_()-x;_(s);
int l=strlen(s),ans=;
if(l<=y){
int d=y-l+;
memset(b,-,d+>>);
for(int i=;i<l;++i)v[s[i]-''].cut(x+i,d);
while(d&)--d,ans+=b[d>>]>>d&;
d>>=;
for(int i=;i<d;++i)ans+=c1[b[i]&]+c1[b[i]>>&]+c1[b[i]>>&]+c1[b[i]>>];
}
printf("%d\n",ans);
}
}
return ;
}

bzoj2628: JZPSTR的更多相关文章

  1. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

随机推荐

  1. Linux学习 -- 用户和用户组管理

    1 用户配置文件 1.1 用户信息文件 /etc/passwd 查看帮助 man 5 passwd -- account:password:UID:GID:GECOS:directory:shell ...

  2. JavaScript(四)---- 函数

    函数主要用来封装具体的功能代码. 函数是由这样的方式进行声明的:关键字 function.函数名.一组参数,以及置于括号中的待执行代码. 格式: function 函数名(形参列表){         ...

  3. 使用for循环还是foreach循环?

    很多时候我们很自然的认为,for循环的时候使用foreach和原来的for循环用下标的方式遍历是相同的. 而且因为foreach循环写法简单,很容易理解,而且少去了很多麻烦的变量,所以估计在学会使用f ...

  4. javascript 手机号码正则表达式验证函数

    代码如下: function checkMobile(){     var sMobile = document.mobileform.mobile.value     if(!(/^1[3|4|5| ...

  5. spring,hibernate配置事务

    1. 新建java project 2. 引入jar 3. src下新建package:com.web.model, com.web.dao, com.web.service, bean.xml 4. ...

  6. Ubuntu和Redhat(Debian)的差别

    这两个最大的区别在包管理模式上. 都是用的Linux核心构架的. Redhat主要集中在 企业级服务器版的制作 是推动LINUX商业化最成功的公司 Redhat对应的桌面版制作 都是由Fedora社区 ...

  7. android 线程池

    http://blog.csdn.net/wangwenhui11/article/details/6760474 http://blog.csdn.net/cutesource/article/de ...

  8. Spring + iBATIS完整示例

    最近研究了一下Spring + iBATIS.发现看别人的例子是一回事,自己写一个完整的应用又是另外一回事.自己受够了网上贴的一知半解的代码. iBATIS是一个持久化框架,封面了sql过程,虽然sq ...

  9. UITableView 性能优化

    网络图片异步加载,SDWebImage. 文字直接 drawInRect/drawAtPoint 绘制,参考 ABTableViewCell,AdvancedTableViewCells. 本地图片也 ...

  10. 认识ExtJS(04)--常见Web框架的ExtJS改造

    常见Web框架的ExtJS改造 怎样开始改造工作? 改造主要集中在对表现层的修改,底层的数据库访问不需要进行任何的变动.也就是说主要集中早这2个方面:一方面是表现部分即是JSP页面:另一方面是客户端与 ...