北京赛区快了,准备袭击数据结构和图论。倒计时 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. 我所认识的EXT2(一)

    前言: 本文是笔者自己在学习文件系统中的一些体会,写出来和大家分享一下.本文首先是介绍了下文件系统的一些理论概念,然后分析了ext2文件系统的原理和部分源码. 文件系统是什么: 人们在认识一件陌生事物 ...

  2. php 扩展 suhosin 配置不当引发的报错及其解决方法

    1. /var/log/messages 频繁报错: Jul :: localhost suhosin[]: ALERT - script tried to increase memory_limit ...

  3. JDOM,dom4j方式解析XML

    <?xml version="1.0" encoding="UTF-8"?> <dataSources> <!-- 定义MySQL ...

  4. 关于wdsl

    WSDL元素 WSDL元素基于XML语法描述了与服务进行交互的基本元素: Type(消息类型):数据类型定义的容器,它使用某种类型系统(如XSD). Message(消息):通信数据的抽象类型化定义, ...

  5. Springboot错误问题总结

    进行springboot+swagger2测试的时候,启动项目发现出现这个问题 把所有的类,配置类都注释掉,不管用,百度搜索之后发现一个解决办法, 半信半疑的加到启动类SpringBootApplic ...

  6. 紫书 习题 8-24 UVa 10366 (构造法)

    又是一道非常复杂的构造法-- #include<cstdio> #include<algorithm> #define REP(i, a, b) for(int i = (a) ...

  7. WPF 内部的5个窗口之 MediaContextNotificationWindow

    原文:WPF 内部的5个窗口之 MediaContextNotificationWindow 本文告诉大家在 WPF 内部的5个窗口的 MediaContextNotificationWindow 是 ...

  8. SWT自定义选项卡CTabFolder

    SWT自定义选项卡CTabFolder 学习了:http://blog.csdn.net/dreajay/article/details/17391731 package com.swt; impor ...

  9. Step by Step Do IOS Swift CoreData Simple Demo

    简单介绍 这篇文章记录了在 IOS 中使用 Swift 操作 CoreData 的一些基础性内容,因为缺乏文档,基本上都是自行实验的结果.错漏不可避免,还请谅解. 部分内容借鉴了 Tim Roadle ...

  10. 用java实现螺旋数组

    接收数组的行数和列数,返回正序和倒序的螺旋数组 package cn.baokx; public class Test { public static void main(String[] args) ...