scoi2010&&bzoj1858序列操作
【题目描述】
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序列操作的更多相关文章
- 【线段树】【P2572】【SCOI2010】序列操作
Description lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 ...
- 【SCOI2010】序列操作
各种繁琐的线段树标记操作...赤裸裸的码农题. 调了一个晚上,最后写篇题解. 题解亮点:代码短,~~跑得慢(连第一页都没挤进去)~~ 其实我跟你们说啊,代码短是好事~~(这里不是说压行好,我的代码不压 ...
- [bzoj1858]序列操作
考虑建立一棵线段树,维护:1.左端点的连续1和:2.右端点的连续1和:3.最长1的连续子序列:4.1的个数:5.将0和1交换后上面的四项:6.懒标记具体实现中,需要注意细节,可以看代码(比较短) 1 ...
- bzoj1858[Scoi2010]序列操作 线段树
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 3079 Solved: 1475[Submit][Statu ...
- BZOJ1858 [Scoi2010]序列操作(线段树)
题目链接 [Scoi2010]序列操作 考验代码能力的一道好题. 思想还是很简单的(直接上线段树),但是比较难写. #include <bits/stdc++.h> using names ...
- 【BZOJ-1858】序列操作 线段树
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 1961 Solved: 991[Submit][Status ...
- bzoj 1858: [Scoi2010]序列操作
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MB 线段树,对于每个区间需要分别维护左右和中间的1和0连续个数,并在op=4时特殊 ...
- BZOJ 1858: [Scoi2010]序列操作( 线段树 )
略恶心的线段树...不过只要弄清楚了AC应该不难.... ---------------------------------------------------------------- #inclu ...
- BZOJ_1858_[Scoi2010]序列操作_线段树
BZOJ_1858_[Scoi2010]序列操作_线段树 Description lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询 ...
随机推荐
- 【转】如何把Json格式字符写进text文件中
http://www.cnblogs.com/insus/p/4306640.html http://json2csharp.chahuo.com/ 本篇一步一步学习怎样把显示于网页的json格式的字 ...
- MySQL 表分区详解MyiSam引擎和InnoDb 区别(实测)
一.什么是表分区通俗地讲表分区是将一大表,根据条件分割成若干个小表.mysql5.1开始支持数据表分区了.如:某用户表的记录超过了1000万条,那么就可以根据入库日期将表分区,也可以根据所在地将表分区 ...
- scala的面向对象编程
1.scala的简单编程 2.构造方法 辅助构造函数是在主构造函数没有的情况下,执行的构造函数. 3.object的介绍 4.半生类和半生对象 5.半生的案例程序(半生类可以调用半生) 6.apply ...
- QStriingList
#include <QCoreApplication> #include<QDebug> #include<QStringList> int main(int ar ...
- CSS之伪类
1. :link 向未被访问的链接添加样式 :visited 向已被访问的链接添加样式 :hover ...
- 【leedcode】 Median of Two Sorted Arrays
https://leetcode.com/problems/median-of-two-sorted-arrays/ There are two sorted arrays nums1 and num ...
- shell中对字符串的处理
1.替换字符串1为字符串2 sed "s/str1/str2/g" 2.获取字符串中的一部分 例:boke-blade 取得boke:sed -e "s/-.*//g&q ...
- sqlserver日志的备份与还原
----------完整备份与还原---------- --完整备份数据库--backup database studb to disk='e:\stu.bak'back ...
- LUA5.3的BNF范式学习笔记
BNF巴科斯范式 {A} 表示 0 或多个 A , [A] 表示一个可选的 A chunk ::= block block ::= {stat} [retstat] stat ::= ‘;’ ...
- PIXHAWK DIY LED扩展板
板载的状态LED灯,因为各种灰机的外壳有可能会被挡住看不到状态.那么我们也是可以用arduino板子来扩展实现外置,其实就是用328P芯片来实现. 这程序支持WS2812B的全彩LED灯. 默认的信号 ...