我太sb啦

合并的时候又漏了,又漏了,又漏了

我个sb

这是个板子题,并不知道为什么SHOI2015会考这么板子的题,但是我又sb了,又sb了,又sb了,又没有1A

显然我是凉了

这道题有三个操作

  1. 区间清零

  2. 将一个区间清零,之后补到另一个区间去,但是有可能补不满

  3. 询问一个区间内最大全零子串

显然这都是线段树的板子操作,对于维护这种区间最长的连续的子串,我们只需要多维护两个数组\(rc[i],lc[i]\),分别表示一个区间内从左开始和从右开始的最长子串,于是就可以合并啦

之后第二个操作看起来很唬人,但是其实也非常板子,我们要找最长的区间,这个区间内的0的个数必须小于等于那个清零了的区间里原来的1的个数

这显然是满足单调性的,我们显然可以将这个区间二分出来

于是这里需要二分一下,同时二分的过程中线段树查询一下一个区间内1的个数来判断,所以操作2的复杂度是\(O(log^2n)\)了

看起来\(O(mlog^2n)\)加上线段树的大常数跑\(200000\)有些危险,但是人要有信仰

之后最sb的是我写错的地方,就是\(pushup\)还有\(query\)合并区间的时候,如果一个区间的左半部分全是0,那么这个区间的从左开始的最长的全0子串肯定可以延伸到右端去,所以这里不能简单地\(lc[i]=lc[i<<1]\)了,我们需要判断一下在合并

可见我有多么sb

代码

  1. #include<iostream>
  2. #include<cstring>
  3. #include<cstdio>
  4. #define re register
  5. #define maxn 200005
  6. #define max(a,b) ((a)>(b)?(a):(b))
  7. struct TREENODE
  8. {
  9. int ll,rr,len;
  10. };
  11. int l[maxn<<2],r[maxn<<2],tag[maxn<<2];
  12. int d[maxn<<2],ans[maxn<<2],lc[maxn<<2],rc[maxn<<2];
  13. int n,m;
  14. inline int read()
  15. {
  16. char c=getchar();
  17. int x=0;
  18. while(c<'0'||c>'9') c=getchar();
  19. while(c>='0'&&c<='9')
  20. x=(x<<3)+(x<<1)+c-48,c=getchar();
  21. return x;
  22. }
  23. inline void pushup(int i)
  24. {
  25. d[i]=d[i<<1]+d[i<<1|1];
  26. if(lc[i<<1]!=r[i<<1]-l[i<<1]+1) lc[i]=lc[i<<1];
  27. else lc[i]=lc[i<<1]+lc[i<<1|1];
  28. if(rc[i<<1|1]!=r[i<<1|1]-l[i<<1|1]+1) rc[i]=rc[i<<1|1];
  29. else rc[i]=rc[i<<1|1]+rc[i<<1];
  30. ans[i]=max(max(ans[i<<1],ans[i<<1|1]),rc[i<<1]+lc[i<<1|1]);
  31. }
  32. void build(int x,int y,int i)
  33. {
  34. l[i]=x;
  35. r[i]=y;
  36. tag[i]=-1;
  37. if(x==y)
  38. {
  39. d[i]=1;
  40. rc[i]=lc[i]=0;
  41. return;
  42. }
  43. int mid=l[i]+r[i]>>1;
  44. build(x,mid,i<<1);
  45. build(mid+1,y,i<<1|1);
  46. pushup(i);
  47. }
  48. inline void pushdown(int i)
  49. {
  50. if(tag[i]==-1) return;
  51. if(tag[i]==1)
  52. {
  53. d[i<<1]=r[i<<1]-l[i<<1]+1;
  54. d[i<<1|1]=r[i<<1|1]-l[i<<1|1]+1;
  55. ans[i<<1|1]=ans[i<<1]=0;
  56. rc[i<<1|1]=rc[i<<1]=0;
  57. lc[i<<1]=lc[i<<1|1]=0;
  58. tag[i<<1|1]=tag[i<<1]=1;
  59. tag[i]=-1;
  60. }
  61. if(!tag[i])
  62. {
  63. d[i<<1|1]=d[i<<1]=0;
  64. rc[i<<1|1]=lc[i<<1|1]=ans[i<<1|1]=r[i<<1|1]-l[i<<1|1]+1;
  65. rc[i<<1]=lc[i<<1]=ans[i<<1]=r[i<<1]-l[i<<1]+1;
  66. tag[i<<1]=tag[i<<1|1]=0;
  67. tag[i]=-1;
  68. }
  69. }
  70. void change(int x,int y,int v,int i)
  71. {
  72. if(x<=l[i]&&y>=r[i])
  73. {
  74. d[i]=v*(r[i]-l[i]+1);
  75. tag[i]=v;
  76. if(!v) lc[i]=rc[i]=ans[i]=r[i]-l[i]+1;
  77. else lc[i]=rc[i]=ans[i]=0;
  78. return;
  79. }
  80. pushdown(i);
  81. int mid=l[i]+r[i]>>1;
  82. if(y<=mid) change(x,y,v,i<<1);
  83. else if(x>mid) change(x,y,v,i<<1|1);
  84. else change(x,y,v,i<<1),change(x,y,v,i<<1|1);
  85. pushup(i);
  86. }
  87. int ask(int x,int y,int i)
  88. {
  89. if(x<=l[i]&&y>=r[i]) return d[i];
  90. pushdown(i);
  91. int mid=l[i]+r[i]>>1;
  92. if(y<=mid) return ask(x,y,i<<1);
  93. if(x>mid) return ask(x,y,i<<1|1);
  94. return ask(x,y,i<<1)+ask(x,y,i<<1|1);
  95. }
  96. TREENODE query(int x,int y,int i)
  97. {
  98. if(x<=l[i]&&y>=r[i]) return (TREENODE){lc[i],rc[i],ans[i]};
  99. pushdown(i);
  100. int mid=l[i]+r[i]>>1;
  101. if(y<=mid) return query(x,y,i<<1);
  102. if(x>mid) return query(x,y,i<<1|1);
  103. TREENODE lson=query(x,y,i<<1),rson=query(x,y,i<<1|1);
  104. int LEN=max(max(lson.len,rson.len),lson.rr+rson.ll);
  105. int LL,RR;
  106. if(lson.ll==mid-l[i]+1) LL=lson.ll+rson.ll;
  107. else LL=lson.ll;
  108. if(rson.rr==r[i]-mid) RR=rson.rr+lson.rr;
  109. else RR=rson.rr;
  110. return (TREENODE){LL,RR,LEN};
  111. }
  112. int main()
  113. {
  114. n=read();
  115. m=read();
  116. build(1,n,1);
  117. int x,y,opt,xx,yy;
  118. while(m--)
  119. {
  120. opt=read();
  121. x=read(),y=read();
  122. if(opt==1) xx=read(),yy=read();
  123. if(!opt) change(x,y,0,1);
  124. if(opt==1)
  125. {
  126. int tot=ask(x,y,1);
  127. if(!tot) continue;
  128. change(x,y,0,1);
  129. int L=1,R=yy-xx+1,ty;
  130. while(L<=R)
  131. {
  132. int mid=L+R>>1;
  133. if(mid-ask(xx,xx+mid-1,1)<=tot) ty=mid,L=mid+1;
  134. else R=mid-1;
  135. }
  136. change(xx,xx+ty-1,1,1);
  137. }
  138. if(opt==2) printf("%d\n",query(x,y,1).len);
  139. }
  140. return 0;
  141. }

【[SHOI2015]脑洞治疗仪】的更多相关文章

  1. 【题解】Luogu P4344 [SHOI2015]脑洞治疗仪

    原题传送门:P4344 [SHOI2015]脑洞治疗仪 前置芝士:珂朵莉树 窝博客里对珂朵莉树的介绍 没什么好说的自己看看吧 珂朵莉树好题啊 我一开始一直Re65 后来重构代码就ac了,或许是rp问题 ...

  2. 【BZOJ4592】[Shoi2015]脑洞治疗仪 线段树

    [BZOJ4592][Shoi2015]脑洞治疗仪 Description 曾经发明了自动刷题机的发明家SHTSC又公开了他的新发明:脑洞治疗仪--一种可以治疗他因为发明而日益增大的脑洞的神秘装置. ...

  3. [SHOI2015]脑洞治疗仪(恶心的线段树,区间最大子段和)

    题目描述: 曾经发明了自动刷题机的发明家 SHTSC 又公开了他的新发明:脑洞治疗仪--一种可以治疗他因为发明而日益增大的脑洞的神秘装置. 为了简单起见,我们将大脑视作一个 01 序列.11代表这个位 ...

  4. bzoj千题计划280:bzoj4592: [Shoi2015]脑洞治疗仪

    http://www.lydsy.com/JudgeOnline/problem.php?id=4592 注意操作1 先挖再补,就是补的范围可以包含挖的范围 SHOI2015 的题 略水啊(逃) #i ...

  5. 洛谷P4344 [SHOI2015]脑洞治疗仪(珂朵莉树)

    传送门 看到区间推倒……推平就想到珂朵莉树 挖脑洞直接assign,填坑先数一遍再assign再暴力填,数数的话暴力数 //minamoto #include<iostream> #inc ...

  6. 【bzoj4592】[Shoi2015]脑洞治疗仪

    由于脑洞的序列不会改变,考虑用线段树维护区间内sum,左边0的个数,右边0的个数,区间内最大脑洞.对于查询l~r最大脑洞可以将l~r分成logn个区间,总复杂度O(nlogn). #include&l ...

  7. 题解 P4344 【[SHOI2015]脑洞治疗仪】

    前言 这道题目呢,看上去很难,实际上我们可以用线段树解决这道题目. 正文 我们维护 sum.len.tag.lmax.rmax.ans. sum 就是这段区间非脑洞的个数 len 就是这段区间的长度 ...

  8. [bzoj4592] [Shoi2015]脑洞治疗仪

    题面无法直视系列. 中规中矩的线段树题. 涉及的操作有:区间赋值为0,计算区间内1的个数,区间赋值为1,求区间内最大的连续的1的个数. #include<cstdio> #include& ...

  9. [SHOI2015]脑洞治疗仪

    嘟嘟嘟 这题其实就是一个线段树维护最大连续和的水题. 别的操作不说,操作1只要二分找区间前\(k\)个0即可. 需要注意的是,因为操作1两区间可能有交,因此要先清空再二分查询-- 复杂度\(O(n l ...

随机推荐

  1. 架构实战项目心得(十一):基于spring-security-oauth2的mysql数据表设计

    一.建立数据库及数据表结构 CREATE SCHEMA IF NOT EXISTS `oauth2` DEFAULT CHARACTER SET utf8 ; USE `oauth2` ; -- -- ...

  2. 1.http请求编程-->基础原理

    一.技术分析 打开网页,不管我们请求的是静态资源还是动态资源,IIS都会根据ISAPI(微软和Process软件公司联合提出的Web服务器上的API标准)这一标准,将请求的文件根据文件后缀名的不同,转 ...

  3. Java 并发(一) --- CAS

    CAS 原理 先来看看下面的代码是否可以输出预期的值.开启了两个线程,是否会输出200 呢 结果由于并发的原因,结果会小于或等于200 , 原因出现在 count++; 由于这一行代码存在三个操作: ...

  4. ​Error -4075: File not found. An error occurred merging module <MODULENAME> for feature <FEATURENAME>.

    利用Install Shield2010制作安装包的时候一直报这样的错误,原以为是我自己安装包制作流程的问题,又重新做了2个,但是还是出现问题. 解决办法: 查找资料发现是Install Shield ...

  5. [javaEE] JDBC快速入门

    JDBC:Java Data Base Connectivity java数据库连接 1.组成JDBC的两个包:主要是接口 java.sql javax.sql 2.相应JDBC的数据库实现 在tom ...

  6. shiro,基于springboot,基于前后端分离,从登录认证到鉴权,从入门到放弃

    这个demo是基于springboot项目的. 名词介绍: ShiroShiro 主要分为 安全认证 和 接口授权 两个部分,其中的核心组件为 Subject. SecurityManager. Re ...

  7. DataTables固定表格宽度(设置横向滚动条)

    当表格的列比较多的时候,可能就需要固定表格的宽度了,默认的100%宽已经不适应了.默认的100%宽要实现改变窗口大小也100%的话,在table元素上添加width="100%", ...

  8. eclipse切换workspace后配置问题

    正常情况下如果切换了eclipse的workspace后,需要重新配置eclipse,但是可以将原工作目录中的.metadata/.plugins/org.eclipse.core.runtime拷贝 ...

  9. 快速数组对象取值与数组映射新数组--array.map

    array.map(callback,[ thisObject]); 1.map方法的作用不难理解,“映射”嘛,也就是原数组被“映射”成对应新数组 a)array.map(()=>值); [1, ...

  10. 医药箱APP静态小项目

    花费了10天时间,纯手写一个医药箱APP静态小项目,里面有上拉加载.左右滑动.弹出层淡入淡出等效果,主要是练习. 以下是一部分页面效果图: 我用的是谷歌的开发者工具的手机端模拟器. 里面需要优化的地方 ...