【题目描述】

lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作:

0 a b 把[a, b]区间内的所有数全变成0

1 a b 把[a, b]区间内的所有数全变成1

2 a b 把[a,b]区间内的所有数全部取反,也就是说把所有的0变成1,把所有的1变成0

3 a b 询问[a, b]区间内总共有多少个1

4 a b 询问[a, b]区间内最多有多少个连续的1

对于每一种询问操作,lxhgww都需要给出回答,聪明的程序员们,你们能帮助他吗?

【输入】

输入数据第一行包括2个数,n和m,分别表示序列的长度和操作数目

第二行包括n个数,表示序列的初始状态

接下来m行,每行3个数,op, a, b,(0<=op<=4,0<=a<=b)

这题真心烦,尤其是对初学线段树的我来说。考试的时候不得不强行写了30分的程序(o(╯□╰)o)。调了一天了,到晚上才调出来,我果然是蒟蒻。

`此处我用的是姜神的思路,又省空间又省时间,真的好。 nod数组代表lazy,如果下面区间数字一致,

可代表 0or1时:update&pushdown 没什么难度。

2时:+一个如果区间不相等,再往下就是了

3时:查询左右即可。

4时:用一个last数组存储前一个区间的值,更新max即可。

#include<cstdio>
#include<cstring>
#include<iostream>
#define push_up if (nod[rt<<1]==nod[rt<<1|1]&&nod[rt]==-1) nod[rt]=nod[rt<<1]
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
using namespace std;
int nod[800005],n,m,last,max1,ans,a;
void init()
{
freopen("operation.in","r",stdin);
freopen("operation.out","w",stdout);
}
void pushdown(int rt)
{
if (nod[rt]!=-1)
{
nod[rt<<1]=nod[rt<<1|1]=nod[rt];
nod[rt]=-1;
}
}
void change(int rt,int l,int r,int a,int b,int color)
{
if (a<=l&&b>=r)
{
nod[rt]=color;
return;
}
pushdown(rt);
int mid=(l+r)>>1;
if (a<=mid) change(lson,a,b,color);
if (b>mid) change(rson,a,b,color);
push_up;
}
void modity(int rt,int l,int r,int a,int b)
{
if (a<=l&&b>=r)
{
if (nod[rt]!=-1)
{
nod[rt]=1-nod[rt];
return;
}
}
pushdown(rt);
int mid=(l+r)>>1;
if (a<=mid) modity(lson,a,b);
if (b>mid) modity(rson,a,b);
push_up;
}
int count(int rt,int l,int r,int a,int b)
{
if (a<=l&&b>=r)
{
if (nod[rt]==1) return r-l+1;
else if (nod[rt]==0)return 0;
}
pushdown(rt);
int mid=(l+r)>>1;
int x1=0;
int x2=0;
if (a<=mid) x1=count(lson,a,b);
if (b>mid) x2=count(rson,a,b);
push_up;
return x1+x2;
}
void find(int rt,int l,int r,int a,int b)
{
if (a<=l&&b>=r)
{
if (nod[rt]!=-1)
{
if (nod[rt]==1&&last==1)
{
max1+=r-l+1;
ans=max(max1,ans);
}
else if (nod[rt]==1&&last==0)
{
max1=r-l+1;
ans=max(max1,ans);
}
else if (nod[rt]==0)
{
max1=0;
}
ans=max(max1,ans);
last=nod[rt];
return;
}
if (l==r) return;
}
pushdown(rt);
int mid=(l+r)>>1;
if (a<=mid) find(lson,a,b);
if (b>mid) find(rson,a,b);
push_up;
}
void work()
{
cin>>n>>m;
int x,y,z;
for (int i=0;i<n;i++)
{
scanf("%d",&a);
change(1,0,n-1,i,i,a);
}
for (int i=1;i<=m;i++)
{
scanf("%d%d%d",&x,&y,&z);
if (x==0||x==1) change(1,0,n-1,y,z,x);
if (x==2) modity(1,0,n-1,y,z);
if (x==3) printf("%d\n",count(1,0,n-1,y,z));
if (x==4)
{
last=1;
max1=ans=0;
find(1,0,n-1,y,z);
printf("%d\n",ans);
}
}
}
int main()
{
init();
work();
return 0;
}

scoi2010&&bzoj1858序列操作的更多相关文章

  1. 【线段树】【P2572】【SCOI2010】序列操作

    Description lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 ...

  2. 【SCOI2010】序列操作

    各种繁琐的线段树标记操作...赤裸裸的码农题. 调了一个晚上,最后写篇题解. 题解亮点:代码短,~~跑得慢(连第一页都没挤进去)~~ 其实我跟你们说啊,代码短是好事~~(这里不是说压行好,我的代码不压 ...

  3. [bzoj1858]序列操作

    考虑建立一棵线段树,维护:1.左端点的连续1和:2.右端点的连续1和:3.最长1的连续子序列:4.1的个数:5.将0和1交换后上面的四项:6.懒标记具体实现中,需要注意细节,可以看代码(比较短) 1 ...

  4. bzoj1858[Scoi2010]序列操作 线段树

    1858: [Scoi2010]序列操作 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 3079  Solved: 1475[Submit][Statu ...

  5. BZOJ1858 [Scoi2010]序列操作(线段树)

    题目链接 [Scoi2010]序列操作 考验代码能力的一道好题. 思想还是很简单的(直接上线段树),但是比较难写. #include <bits/stdc++.h> using names ...

  6. 【BZOJ-1858】序列操作 线段树

    1858: [Scoi2010]序列操作 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 1961  Solved: 991[Submit][Status ...

  7. bzoj 1858: [Scoi2010]序列操作

    1858: [Scoi2010]序列操作 Time Limit: 10 Sec  Memory Limit: 64 MB 线段树,对于每个区间需要分别维护左右和中间的1和0连续个数,并在op=4时特殊 ...

  8. BZOJ 1858: [Scoi2010]序列操作( 线段树 )

    略恶心的线段树...不过只要弄清楚了AC应该不难.... ---------------------------------------------------------------- #inclu ...

  9. BZOJ_1858_[Scoi2010]序列操作_线段树

    BZOJ_1858_[Scoi2010]序列操作_线段树 Description lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询 ...

随机推荐

  1. SQL INSERT INTO 语句

    SQL Order By SQL update INSERT INTO 语句 INSERT INTO 语句用于向表格中插入新的行. 语法 INSERT INTO 表名称 VALUES (值1, 值2, ...

  2. 启动Hive时出现的问题

    Caused by: org.apache.hadoop.hive.ql.metadata.HiveException: java.lang.RuntimeException: Unable to i ...

  3. centos 6.5安装node.js

    1.检查是否安装gcc编译器 rpm -q gcc rpm -q gcc-c++ 2.如果没有安装则通过以下代码安装gcc编译器 yum -y install gcc-c++ kernel-devel ...

  4. 手势响应 ,避免点击多个cell同时响应同一手势多次,只响应第一个cell

    http://www.cnblogs.com/wfwenchao/articles/3700205.html UIView除了负责展示内容给用户外还负责响应用户事件.本章主要介绍UIView用户交互相 ...

  5. SVG知识难点

      参考资料:http://www.w3cplus.com/css3/clip.html   1.clip:默认值是auto,为不裁剪 <img border="0" src ...

  6. maven常用插件pom配置

    一.问题描述: 部署一个maven打包项目时,jar包,依赖lib包全部手动上传至服务器,然后用maven部署报错:Exception in thread "main" java. ...

  7. ssh 配合 tar 实现远程推送

    tar命令和ssh配合使用 如何在空间不是很富裕的情况,把文件从一个分区tar到另外一个分区,其实还有很多办法的,使用管道命令就可以实现 如: #tar -cvf home |(cd /datavg3 ...

  8. using的作用

    using (ServiceHost host = new ServiceHost(typeof(OperationService))) 这样写代码有何好处?using的作用范围自动回收垃圾,在这个范 ...

  9. Java 高精度数字

    BigInteger // 高精度整数 BigDecimal //高精度小数  小数位数不受限制

  10. 用Canvas制作剪纸效果

    在做剪纸效果之前,先介绍剪纸效果运用到的一些知识: 1.阴影: 在Canvas之中进行绘制时,可以通过修改绘图环境中的如下4个属性值来指定阴影效果: shadowColor:CSS格式的颜色字串.默认 ...