这个是去年遗留历史问题,之前思路混乱,搞了好多发都是WA,就没做了

自从上次做了大白书上那个双重懒惰标记的题目,做这个就思路很清晰了

跟上次大白上那个差不多,这个也是有一个sets标记,代表这个区间全部置为0或者1,没有置位的时候为-1

还有个rev标记,代表翻转操作,0代表当前不翻,1代表当前翻

要注意一下优先级,发现有不同的弄法,我是这个弄得,有set操作的时候,set标记设值,并把当前节点的rev标记设为0,因为不管要不要rev,当前set操作肯定直接覆盖了

rev操作不改变set操作,在pushdown的时候,先考虑set标记再弄rev标记,这也是很好理解的,因为一旦rev和set共存,肯定是rev在set后面。

有几个细节要注意一下,一开始行云流水一气呵成,发现还是WA了,就是这几个地方,

1.除了set的时候强制给rev弄成0,其他任何时候对rev标记操作都是^1,取反,这个也很好理解,之前在pushdown里面我就是直接传值,肯定不对嘛

2.在pushdown里面的set下传操作也要记得把子树的rev标记抹除,一开始只在主修改函数里写了,这里没写,WA的不明不白,理由跟上面的一样

然后就基本上没问题了

#include <iostream>
#include <cstdio>
#include <cstring>
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
using namespace std;
const int N=100000+10;
int d[N*3],maxc[N*3],lc[N*3],rc[N*3],maxc0[N*3],lc0[N*3],rc0[N*3];
int sets[N*3],rev[N*3];
int n,Q;
int A[N];
void up(int rt,int l,int r)
{
int mid=(l+r)>>1;
d[rt]=d[rt<<1]+d[rt<<1|1];
maxc[rt]=max(maxc[rt<<1],maxc[rt<<1|1]);
maxc[rt]=max(maxc[rt],lc[rt<<1|1]+rc[rt<<1]);
lc[rt]=lc[rt<<1];
rc[rt]=rc[rt<<1|1];
if (lc[rt<<1]==mid-l+1) lc[rt]+=lc[rt<<1|1];
if (rc[rt<<1|1]==r-mid) rc[rt]+=rc[rt<<1]; maxc0[rt]=max(maxc0[rt<<1],maxc0[rt<<1|1]);
maxc0[rt]=max(maxc0[rt],lc0[rt<<1|1]+rc0[rt<<1]);
lc0[rt]=lc0[rt<<1];
rc0[rt]=rc0[rt<<1|1];
if (lc0[rt<<1]==mid-l+1) lc0[rt]+=lc0[rt<<1|1];
if (rc0[rt<<1|1]==r-mid) rc0[rt]+=rc0[rt<<1];
}
void build(int rt,int l,int r)
{
sets[rt]=-1;
rev[rt]=0;
if (l>=r){
d[rt]=maxc[rt]=A[l];
lc[rt]=rc[rt]=A[l];
lc0[rt]=rc0[rt]=maxc0[rt]=1-A[l];
return;
}
int mid=(l+r)>>1;
build(lson);
build(rson);
up(rt,l,r);
}
void pushdown(int rt,int l,int r)
{
if (l>=r) return;
int mid=(l+r)>>1;
if (sets[rt]>=0){
maxc[rt<<1]=lc[rt<<1]=rc[rt<<1]=d[rt<<1]=(mid-l+1)*sets[rt];
maxc[rt<<1|1]=lc[rt<<1|1]=rc[rt<<1|1]=d[rt<<1|1]=(r-mid)*sets[rt]; maxc0[rt<<1]=lc0[rt<<1]=rc0[rt<<1]=(mid-l+1)*(1-sets[rt]);
maxc0[rt<<1|1]=lc0[rt<<1|1]=rc0[rt<<1|1]=(r-mid)*(1-sets[rt]); sets[rt<<1]=sets[rt<<1|1]=sets[rt];
rev[rt<<1]=rev[rt<<1|1]=0;
sets[rt]=-1; }
if (rev[rt]>0){
d[rt<<1]=(mid-l+1)-d[rt<<1];
d[rt<<1|1]=(r-mid)-d[rt<<1|1]; int t1,t2,t3;
t1=maxc[rt<<1];t2=lc[rt<<1];t3=rc[rt<<1];
maxc[rt<<1]=maxc0[rt<<1];
lc[rt<<1]=lc0[rt<<1];
rc[rt<<1]=rc0[rt<<1];
maxc0[rt<<1]=t1;
lc0[rt<<1]=t2;
rc0[rt<<1]=t3;
t1=maxc[rt<<1|1];t2=lc[rt<<1|1];t3=rc[rt<<1|1];
maxc[rt<<1|1]=maxc0[rt<<1|1];
lc[rt<<1|1]=lc0[rt<<1|1];
rc[rt<<1|1]=rc0[rt<<1|1];
maxc0[rt<<1|1]=t1;
lc0[rt<<1|1]=t2;
rc0[rt<<1|1]=t3;
rev[rt<<1]^=1;
rev[rt<<1|1]^=1;
rev[rt]=0;
}
}
void change(int val,int L,int R,int rt,int l,int r)
{
if (L<=l && r<=R){
d[rt]=(r-l+1)*val;
lc[rt]=rc[rt]=(r-l+1)*val;
maxc[rt]=(r-l+1)*val;
lc0[rt]=rc0[rt]=maxc0[rt]=(r-l+1)*(1-val);
sets[rt]=val;
rev[rt]=0;
return;
}
pushdown(rt,l,r);
int mid=(l+r)>>1;
if (L>mid) change(val,L,R,rson);
else
if (R<=mid) change(val,L,R,lson);
else{
change(val,L,R,rson);
change(val,L,R,lson);
}
up(rt,l,r);
}
void revers(int L,int R,int rt,int l,int r)
{ if (L<=l && r<=R)
{ d[rt]=(r-l+1)-d[rt];
int t1,t2,t3;
t1=maxc[rt];t2=lc[rt];t3=rc[rt];
maxc[rt]=maxc0[rt];lc[rt]=lc0[rt];rc[rt]=rc0[rt];
maxc0[rt]=t1;lc0[rt]=t2;rc0[rt]=t3;
rev[rt]^=1;
return;
}
pushdown(rt,l,r);
int mid=(l+r)>>1;
if (R<=mid) revers(L,R,lson);
else
if (L>mid) revers(L,R,rson);
else
{
revers(L,R,lson);
revers(L,R,rson);
}
up(rt,l,r);
}
int output(int op,int L,int R,int rt,int l,int r)
{
if (L==l && r==R){
if (op==3) return d[rt];
else return maxc[rt];
}
pushdown(rt,l,r);
int mid=(l+r)>>1;
if (L>mid) return output(op,L,R,rson);
else
if (R<=mid) return output(op,L,R,lson);
else
{
int ret1=output(op,L,mid,lson);
int ret2=output(op,mid+1,R,rson);
if (op==3) return ret1+ret2;
else{
int ret=max(ret1,ret2);
int t1=min(rc[rt<<1],mid-L+1);
int t2=min(lc[rt<<1|1],R-mid);
ret=max(ret,t1+t2);
return ret;
}
}
}
int main()
{
int t,op,a,b;
scanf("%d",&t);
while (t--)
{
scanf("%d%d",&n,&Q);
for (int i=1;i<=n;i++) scanf("%d",&A[i]);
build(1,1,n);
while (Q--)
{
scanf("%d%d%d",&op,&a,&b);
a++;b++;
if (op<=1){
change(op,a,b,1,1,n);
}
else
if (op==2){
revers(a,b,1,1,n);
}
else
{
int ans=output(op,a,b,1,1,n);
printf("%d\n",ans);
}
}
}
return 0;
}

  

HDU 3397 线段树 双懒惰标记的更多相关文章

  1. hdu 3397 线段树双标记

    Sequence operation Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Othe ...

  2. HDU 4107 Gangster(线段树 特殊懒惰标记)

    两种做法. 第一种:标记区间最大值和最小值,若区间最小值>=P,则本区间+2c,若区间最大值<P,则本区间+c.非常简单的区间更新. 最后发一点牢骚:最后query查一遍就行,我这个2B竟 ...

  3. poj3468 线段树的懒惰标记

    题目链接:poj3468 题意:给定一段数组,有两种操作,一种是给某段区间加c,另一种是查询一段区间的和 思路:暴力的方法是每次都给这段区间的点加c,查询也遍历一遍区间,复杂度是n*n,肯定过不去,另 ...

  4. POJ 3667 & HDU 3308 & HDU 3397 线段树的区间合并

    看到讲课安排上 线段树有一节课"区间合并" 我是迷茫的 因为并没有见过 然后了解了一下题目 发现以前写过 还是很麻烦的树链剖分 大概是 解决带修改的区间查询"连续问题&q ...

  5. HDU 3954 Level up (线段树特殊懒惰标记)

    http://blog.csdn.net/acm_cxlove/article/details/7548087 感觉最巧的是定义了min_dis……将区间内有无英雄升级分开处理 #include &l ...

  6. hdu 3397 线段树

    题意: Change operations:0 a b change all characters into '0's in [a , b]1 a b change all characters in ...

  7. HDU 3397 线段树区间修改

    Sequence operation Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Othe ...

  8. hdu 4578 线段树(标记处理)

    Transformation Time Limit: 15000/8000 MS (Java/Others)    Memory Limit: 65535/65536 K (Java/Others) ...

  9. hdu 3954 线段树 (标记)

    Level up Time Limit: 10000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total ...

随机推荐

  1. Eclipse传递main函数参数

    在项目上右击 Run As->Run Configurations...->Arguments->在Program arguments:的文本框中输入你要传入的参数,若有几个参数则在 ...

  2. vue cli3.0打包

    1.vue cli3.0需要在项目根目录下配置webpack  包括反向代理以及打包文件路径 const webpack = require("webpack"); module. ...

  3. VS2010如何进行程序调试

    VS2010如何进行程序调试 一.前言 对于初步学习C++的朋友,程序的调试是一项必备的技能.尤其是像C++这样难学的语言,程序调试的基本方法更是至关重要.毕竟,谁也不想自己幸幸苦苦一天敲出来的代码就 ...

  4. 《React后台管理系统实战 :四》产品分类管理页:添加产品分类、修改(更新)产品分类

    一.静态页面 目录结构 F:\Test\react-demo\admin-client\src\pages\admin\category add-cate-form.jsx index.jsx ind ...

  5. Pytorch【直播】2019 年县域农业大脑AI挑战赛---初级准备(一)切图

    比赛地址:https://tianchi.aliyun.com/competition/entrance/231717/introduction 这次比赛给的图非常大5万x5万,在训练之前必须要进行数 ...

  6. 新闻网大数据实时分析可视化系统项目——6、HBase分布式集群部署与设计

    HBase是一个高可靠.高性能.面向列.可伸缩的分布式存储系统,利用Hbase技术可在廉价PC Server上搭建 大规模结构化存储集群. HBase 是Google Bigtable 的开源实现,与 ...

  7. webpack配置自动打包重新运行npm run dev出现报错

    webpack配置自动打包重新运行npm run dev出现报错 运行npm run dev出现如下报错 Listening at http://localhost:8080(node:2328) U ...

  8. 「SP1716」GSS3 - Can you answer these queries III

    传送门 Luogu 解题思路 区间最大子段和板子题. 考虑用线段树来做. 对于一个线段树节点所包含区间,它的最大子段和有两种情况,包含中点与不包含. 不包含的情况直接从左右子树转移. 对于包含的情况: ...

  9. 1-8SpringBoot之切面AOP

    SpringBoot提供了强大AOP支持,我们前面讲解过AOP面向切面,所以这里具体AOP原理就补具体介绍: AOP切面主要是切方法,我们一般搞一些日志分析和事务操作,要用到切面,类似拦截器: @As ...

  10. python2学习------基础语法4(模块)

    1.整体结构层次(a.py,b.py) 目标:b.py文件中导入a.py里面定义的class A,并调用A类里面的属性或相关方法. 2.模块导入 <1>  __init__.py < ...