北京赛区快了,准备袭击数据结构和图论。倒计时 18天,线段树区间合并。维护一个最长连续。。

题意:给一个01串,以下有一些操作,问区间最长的连续的1的个数

思路:非常裸的线段树区间合并
#include<iostream>
#include<cstdio>
#include<map>
#include<set>
#include<cmath>
#define lson id << 1
#define rson id << 1|1
using namespace std;
const int M = 1e6+8;
int a[M];
struct tree{
int l,r;
int rsum1,lsum1,msum1,lsum0,rsum0,msum0;
int flag;
int mid(){
return (l+r)/2;
}
}node[M];
void pushdown(int id){
if(node[id].flag){
node[lson].flag ^= 1;
node[rson].flag ^= 1;
node[id].flag = 0;
swap(node[lson].lsum1,node[lson].lsum0);
swap(node[lson].rsum1,node[lson].rsum0);
swap(node[lson].msum1,node[lson].msum0); swap(node[rson].lsum1,node[rson].lsum0);
swap(node[rson].rsum1,node[rson].rsum0);
swap(node[rson].msum1,node[rson].msum0);
}
}
void pushup(int id){
int ll = node[lson].r-node[lson].l + 1;
int rl = node[rson].r-node[rson].l + 1;
node[id].lsum1 = node[lson].lsum1;
if(node[lson].lsum1 == ll)node[id].lsum1 += node[rson].lsum1;
node[id].rsum1 = node[rson].rsum1;
if(node[rson].rsum1 == rl)node[id].rsum1 += node[lson].rsum1;
node[id].msum1 = max(max(node[rson].msum1,node[lson].msum1),node[lson].rsum1+node[rson].lsum1); node[id].lsum0 = node[lson].lsum0;
if(node[lson].lsum0 == ll)node[id].lsum0 += node[rson].lsum0;
node[id].rsum0 = node[rson].rsum0;
if(node[rson].rsum0 == rl)node[id].rsum0 += node[lson].rsum0;
node[id].msum0 = max(max(node[rson].msum0,node[lson].msum0),node[lson].rsum0+node[rson].lsum0); }
void build(int l,int r,int id){
node[id].l = l;
node[id].r = r;
node[id].flag = 0;
if(l == r){
if(a[l] == 1){
node[id].lsum1 = node[id].rsum1 = node[id].msum1 = 1;
node[id].lsum0 = node[id].rsum0 = node[id].msum0 = 0;
}else{
node[id].lsum1 = node[id].rsum1 = node[id].msum1 = 0;
node[id].lsum0 = node[id].rsum0 = node[id].msum0 = 1;
}
return;
}
int mid = node[id].mid();
build(l,mid,lson);
build(mid+1,r,rson);
pushup(id);
}
void update(int id,int l,int r){
if(node[id].l == l && node[id].r == r){
node[id].flag ^= 1;
swap(node[id].lsum1,node[id].lsum0);
swap(node[id].rsum1,node[id].rsum0);
swap(node[id].msum1,node[id].msum0);
return ;
}
pushdown(id);
int mid = node[id].mid();
if(r <= mid)update(lson,l,r);
else if(l>mid)update(rson,l,r);
else {
update(lson,l,mid);
update(rson,mid+1,r);
}
pushup(id);
}
int query(int id,int l,int r){
if(node[id].l == l && node[id].r ==r){
return node[id].msum1;
}
pushdown(id);
int mid = node[id].mid();
if(r <=mid)return query(lson,l,r);
else
if(l > mid)return query(rson,l,r);
else {
int ll = query(lson,l,mid);
int rr = query(rson,mid+1,r);
int a = node[lson].rsum1;
if(a > (node[lson].r - l +1))a = node[lson].r - l +1 ; int b = node[rson].lsum1;
if(b > (r - node[rson].l+1))b = r - node[rson].l+1;
return max(max(ll,rr),a+b);
} }
int main(){
int n,m,op,l,r;
while(~scanf("%d",&n)){
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
build(1,n,1);
scanf("%d",&m);
while(m--){
scanf("%d%d%d",&op,&l,&r);
if(!op)printf("%d\n",query(1,l,r));
else update(1,l,r);
}
}
}

HDU 3911 线段树区间合并的更多相关文章

  1. HDU 3911 线段树区间合并、异或取反操作

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...

  2. hdu 3308(线段树区间合并)

    LCIS Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  3. hdu 1806(线段树区间合并)

    Frequent values Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)T ...

  4. hdu 3308 线段树 区间合并+单点更新+区间查询

    LCIS Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  5. hdu 3911 Black And White (线段树 区间合并)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3911 题意: 给你一段01序列,有两个操作: 1.区间异或,2.询问区间最长的连续的1得长度 思路: ...

  6. HDU 3911 Black And White(线段树区间合并+lazy操作)

    开始以为是水题,结果...... 给你一些只有两种颜色的石头,0为白色,1为黑色. 然后两个操作: 1 l r 将[ l , r ]内的颜色取反 0 l r 计算[ l , r ]内最长连续黑色石头的 ...

  7. HDU 3308 LCIS (线段树区间合并)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3308 题目很好懂,就是单点更新,然后求区间的最长上升子序列. 线段树区间合并问题,注意合并的条件是a[ ...

  8. HDU 3308 (线段树区间合并)

    http://acm.hdu.edu.cn/showproblem.php?pid=3308 题意: 两个操作  : 1 修改 单点  a 处的值. 2 求出 区间[a,b]内的最长上升子序列. 做法 ...

  9. HDU 6638 - Snowy Smile 线段树区间合并+暴力枚举

    HDU 6638 - Snowy Smile 题意 给你\(n\)个点的坐标\((x,\ y)\)和对应的权值\(w\),让你找到一个矩形,使这个矩阵里面点的权值总和最大. 思路 先离散化纵坐标\(y ...

随机推荐

  1. 虚拟机创建后该如何获取IP地址并访问互联网实用教程

    之前在做项目的时候主机IP地址.网关.DNS.子网掩码等都是公司或者对方直接给提供的,但是如果我们自己想搭建一台虚拟机或者一台集群的话,手头又没有IP地址,该肿么办呢?   白慌,这里介绍一个小技巧, ...

  2. python中*号用法总结

    python 中有很多地方用到星号,有时候会想知道这个*是干嘛用的,总结如下,有不当之处,还望不吝指出,谢谢.1.乘法: 在很多时候是用作乘法的,例如: In [90]: 2*7 Out[90]: 1 ...

  3. 微软的鼠标 Microsoft mouse

    微软是做软件出身的厂商, 所以微软开发的软件质量毋庸置疑,Windows操作系统还有诸如Office的办公软件拥有庞大的用户群. 微软家的Visual Studio也被号称宇宙最强IDE,我个人也每天 ...

  4. js验证输入框

    项目开发中,都会有对表单form中的input输入框进行一些限制验证 如: <!DOCTYPE html><html lang="en"> <head ...

  5. HDU-1358 Period 字符串问题 KMP算法 求最小循环节

    题目链接:https://cn.vjudge.net/problem/HDU-1358 题意 给一个字符串,对下标大于2的元素,问有几个最小循环节 思路 对每个元素求一下minloop,模一下就好 提 ...

  6. Nginx 安装 自用

    hostnamectl set-hostname nginx systemctl stop firewalld.service systemctl disable firewalld.service ...

  7. AWS中国EC2 公网IP登录免pemKEY修改shh 配置文件

    个人使用记录 1:KEY 授权 chmod 400 VPN.pem 2:连接 ssh -i "VPN.pem" ubuntu@ec2-54-183-119-93.us-west-1 ...

  8. Linux学习总结(10)——Linux查看CPU和内存使用情况

    在系统维护的过程中,随时可能有需要查看 CPU 使用率,并根据相应信息分析系统状况的需要.在 CentOS 中,可以通过 top 命令来查看 CPU 使用状况.运行 top 命令后,CPU 使用状态会 ...

  9. ZOJ 3369 Saving Princess

    Saving Princess Time Limit: 1000ms Memory Limit: 32768KB This problem will be judged on ZJU. Origina ...

  10. hadoop 2.6.0 LightWeightGSet源码分析

    LightWeightGSet的作用用一个数组来存储元素,而且用链表来解决冲突.不能rehash.所以内部数组永远不用改变大小.此类不支持空元素. 此类也不是线程安全的.有两个类型參数.第一个用于查找 ...