北京赛区快了,准备袭击数据结构和图论。倒计时 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. C++ 获取某一文件夹下的所有文件名

    //********************************************************************** // Method: 获取文件夹下所有文件 // Fu ...

  2. python中index、slice与slice assignment用法

    python中index.slice与slice assignment用法 一.index与slice的定义: index用于枚举list中的元素(Indexes enumerate the elem ...

  3. 人工智能,你到底是天使or魔鬼?

    人工智能的概念早在60多年前就被提出,但又一度沉寂.随着谷歌人工智能程序AlphaGo(阿尔法狗)战胜围棋世界冠军李世石,再次为世人瞩目.然而,与无限风光一起相伴而来的,还有关于人工智能的种种争议! ...

  4. tab栏切换

    最简单的tab栏切换 html部分 <ul class="tab"> <li class="item">待支付(1)</li> ...

  5. Vue中两种跳转方式

    第一种:通过标签跳转,<router-link></router-link> 第二种:通过js跳转,定义点击事件进行跳转

  6. php7 memcache和memcached.so扩展

    php7安装memcache和memcached扩展 https://github.com/websupport-sk/pecl-memcache https://github.com/php-mem ...

  7. 【图灵杯 E也即POJ 3368】简单的RMQ

    Description 给定一个数组,其中的元素满足非递减顺序.任意给定一个区间[i,j],求其中某个元素重复出现的最大次数. Input 多组数据输入.每组数据的第一行包含两个整数n和q(1< ...

  8. 2015 Multi-University Training Contest 2 hdu 5306 Gorgeous Sequence

    Gorgeous Sequence Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Othe ...

  9. COGS——T 1578. 次小生成树初级练习题

    http://www.cogs.pro/cogs/problem/problem.php?pid=1578 ☆   输入文件:mst2.in   输出文件:mst2.out   简单对比时间限制:1 ...

  10. JavaScript(14)jQuery(JavaScript 库)

    JavaScript 框架(库) JavaScript 高级程序设计(特别是对浏览器差异的复杂处理),通常非常困难也非常耗时.为了应对这些调整,很多的 JavaScript (helper) 库应运而 ...