Problem Description
There are a bunch of stones on the beach; Stone color is white or black. Little Sheep has a magic brush, she can change the color of a continuous stone, black to white, white to black. Little Sheep like black very much, so she want to know the longest period
of consecutive black stones in a range [i, j].
 
Input
  There are multiple cases, the first line of each case is an integer n(1<= n <= 10^5), followed by n integer 1 or 0(1 indicates black stone and 0 indicates white stone), then is an integer M(1<=M<=10^5) followed by M operations formatted as x i j(x = 0 or
1) , x=1 means change the color of stones in range[i,j], and x=0 means ask the longest period of consecutive black stones in range[i,j]
 
Output
When x=0 output a number means the longest length of black stones in range [i,j].
 
Sample Input
4
1 0 1 0
5
0 1 4
1 2 3
0 1 4
1 3 3
0 4 4
 
Sample Output
1
2
0

分段树依旧是难题,本题能够说是分段树的基本操作,可是由于情况好多,程序好长,所以十分耗时间。

之所以使用线段树,不使用一般的动态规划法,是要把每次查询的时间效率降到 (Lgn)。

分段,每段都要维护8个数据,所以非常繁琐。

做的好累,代码改动了非常多次,终于代码算比較清晰的了。有分段树思想的人应该都不难follow。

Hdu是不同意使用free释放空间的,否则出错,奇怪的OJ。

#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <math.h>
using namespace std; class BlackAndWhite3911
{
int arrSize, tSize;
int *arr;
struct Node
{
int le0, le1, ri0, ri1;
int len0, len1;
int totalLen;
bool lazy;
}; Node *SegTree;
void updateNode(int r, int L, int R)
{
if (SegTree[L].le0 == SegTree[L].totalLen)
SegTree[r].le0 = SegTree[L].le0 + SegTree[R].le0;
else SegTree[r].le0 = SegTree[L].le0; if (SegTree[R].ri0 == SegTree[R].totalLen)
SegTree[r].ri0 = SegTree[L].ri0 + SegTree[R].ri0;
else SegTree[r].ri0 = SegTree[R].ri0; if (SegTree[L].le1 == SegTree[L].totalLen)
SegTree[r].le1 = SegTree[L].le1 + SegTree[R].le1;
else SegTree[r].le1 = SegTree[L].le1; if (SegTree[R].ri1 == SegTree[R].totalLen)
SegTree[r].ri1 = SegTree[L].ri1 + SegTree[R].ri1;
else SegTree[r].ri1 = SegTree[R].ri1; int a = max(SegTree[L].len0, SegTree[R].len0);
int b = SegTree[L].ri0 + SegTree[R].le0;
SegTree[r].len0 = max(a, b); a = max(SegTree[L].len1, SegTree[R].len1);
b = SegTree[L].ri1 + SegTree[R].le1;
SegTree[r].len1 = max(a, b);
} void conHelper(int low, int up, int r = 0)
{
if (low == up)
{
if (0 == arr[low])
{
SegTree[r].len0 = 1, SegTree[r].len1 = 0;
SegTree[r].le0 = 1, SegTree[r].ri0 = 1;
SegTree[r].le1 = 0, SegTree[r].ri1 = 0;
}
else
{
SegTree[r].len0 = 0, SegTree[r].len1 = 1;
SegTree[r].le0 = 0, SegTree[r].ri0 = 0;
SegTree[r].le1 = 1, SegTree[r].ri1 = 1;
}
SegTree[r].lazy = false;
SegTree[r].totalLen = 1;
return ;
} int mid = low + ((up-low)>>1);
int le = (r<<1) + 1;
int ri = (r<<1) + 2; conHelper(low, mid, le);
conHelper(mid+1, up, ri); SegTree[r].totalLen = up - low + 1;
updateNode(r, le, ri);
SegTree[r].lazy = false;
} void conTree()
{
int h = (int) ceil(log((double)arrSize)/log(2.0)) + 1;
tSize = (int) pow(2.0, h) - 1;
SegTree = (Node *) malloc(sizeof(Node) * tSize);
conHelper(0, arrSize-1);
} void accessNode(int r)
{
SegTree[r].lazy = !SegTree[r].lazy;
swap(SegTree[r].le0, SegTree[r].le1);
swap(SegTree[r].ri0, SegTree[r].ri1);
swap(SegTree[r].len0, SegTree[r].len1);
} void segUpdate(const int low, const int up, int L, int R, int r = 0)
{
if (low == L && R == up)
{
accessNode(r);
return;
} int le = (r<<1) + 1;
int ri = (r<<1) + 2; if (SegTree[r].lazy)
{
SegTree[r].lazy = false;
if (le < tSize) accessNode(le);
if (ri < tSize) accessNode(ri);
} int M = L + ((R-L)>>1); if (up <= M) segUpdate(low, up, L, M, le);
else if (low > M) segUpdate(low, up, M+1, R, ri);
else
{
segUpdate(low, M, L, M, le);
segUpdate(M+1, up, M+1, R, ri);
}
updateNode(r, le, ri);
} int getLongest(const int low, const int up, int L, int R, int r = 0)
{
if (low == L && R == up)//不能low <= L && R <= up
{
return SegTree[r].len1;
} int le = (r<<1) + 1;
int ri = (r<<1) + 2;
if (SegTree[r].lazy)
{
SegTree[r].lazy = false;
if (le < tSize) accessNode(le);
if (ri < tSize) accessNode(ri);
} int M = L + ((R-L)>>1); //在任一边子树
if (up <= M) return getLongest(low, up, L, M, le);
if (low > M) return getLongest(low, up, M+1, R, ri); //跨左右子树的情况
int llen = getLongest(low, M, L, M, le);//(low, up, L, M, le);
int rlen = getLongest(M+1, up, M+1, R, ri);//(low, up, M+1, R, ri); int a = min(M-low+1, SegTree[le].ri1);
int b = min(up-M, SegTree[ri].le1);
int c = a + b; return max(c, max(llen, rlen));
}
public:
BlackAndWhite3911()
{
int N;
while ( (scanf("%d", &N) != EOF))
{
arrSize = N;
arr = (int *)malloc(sizeof(int) * arrSize); for (int i = 0; i < N; i++)
{
scanf("%d", &arr[i]);
} conTree(); int T;
scanf("%d", &T);
int a, b, c;
while (T--)
{
scanf("%d %d %d", &a, &b, &c);
if (0 == a)
printf("%d\n",getLongest(b-1, c-1, 0, arrSize-1));
else segUpdate(b-1, c-1, 0, arrSize-1);
}
}
} ~BlackAndWhite3911()
{
if (arr) free(arr);
if (SegTree) free(SegTree);
}
};

HDU 3911 Black And White 分段树 题解的更多相关文章

  1. hdu 3911 Black And White(线段树)

    题目连接:hdu 3911 Black And White 题目大意:给定一个序列,然后有M次操作: 0 l r:表示询问l,r中最大连续1的个数 1 l r:表示将l,r区间上的数取反 解题思路:线 ...

  2. HDU 3911 Black and White (线段树,区间翻转)

      [题目地址] vjudge HDU [题目大意] 海滩上有一堆石头. 石头的颜色是白色或黑色. 小肥羊拥有魔术刷,她可以改变连续石的颜色,从黑变白,从白变黑. 小肥羊非常喜欢黑色,因此她想知道范围 ...

  3. HDU 3911 Black And White (线段树区间合并 + lazy标记)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3911 给你n个数0和1,m个操作: 0操作  输出l到r之间最长的连续1的个数 1操作  将l到r之间 ...

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

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

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

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

  6. HDU 3911 Black And White

    Black And White Time Limit: 3000ms Memory Limit: 32768KB This problem will be judged on HDU. Origina ...

  7. HDU 1556 Color the Ball 线段树 题解

    本题使用线段树自然能够,由于区间的问题. 这里比較难想的就是: 1 最后更新须要查询全部叶子节点的值,故此须要使用O(nlgn)时间效率更新全部点. 2 截取区间不能有半点差错.否则答案错误. 这两点 ...

  8. HDU 6089 Rikka with Terrorist (线段树)

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=6089 题解 这波强行维护搞得我很懵逼... 扫描线,只考虑每个点能走到左上方(不包括正上方,但包括正左 ...

  9. R - Weak Pair HDU - 5877 离散化+权值线段树+dfs序 区间种类数

    R - Weak Pair HDU - 5877 离散化+权值线段树 这个题目的初步想法,首先用dfs序建一颗树,然后判断对于每一个节点进行遍历,判断他的子节点和他相乘是不是小于等于k, 这么暴力的算 ...

随机推荐

  1. 京东金融集团BD部门招聘 BD经理

    新标签页http://74.55.154.136/ 互联网招聘_cnBeta.COM 北京 / 全职 / 20k-30k / 经验3-5年 / 本科及以上 / 1天前发布 职位诱惑 : 五险一金 职位 ...

  2. 浙江大学PAT上机题解析之3-04. 一元多项式的乘法与加法运算

    设计函数分别求两个一元多项式的乘积与和. 输入格式说明: 输入分2行,每行分别先给出多项式非零项的个数,再以指数递降方式输入一个多项式非零项系数和指数(绝对值均为不超过1000的整数).数字间以空格分 ...

  3. BZOJ 1101: [POI2007]Zap( 莫比乌斯反演 )

    求 answer = ∑ [gcd(x, y) = d] (1 <= x <= a, 1 <= y <= b) . 令a' = a / d, b' = b / d, 化简一下得 ...

  4. API通用设计原则

    什么是好的API? ·        完备(Be Complete) 对确定重点支持的用户场景具有完备的功能支持.就是说,用户通过对一组API的调用能够完成预期的功能. ·        不冗余(Be ...

  5. 基于visual Studio2013解决面试题之0808寻找中间数

     题目

  6. asp.net2.0安全性(2)--用户个性化设置(2)--转载来自车老师

    上一篇我们用Profile.age等方式可以读取用户的年龄和其它的信息,但有的时候我们要查询显示所有用户的信息,但asp.net没有提供查询所有用户信息的功能,我们只能对现有的用户逐一查询其Profi ...

  7. 【Tips】Endnote导入IEEE Xplore文献方法《转载》

    1. 在IEEE XPlore中点击“Download Citation”: 2. 选中“Citation & Abstract”和“EndNote,Procite,RefMan”两个选项: ...

  8. Eclipse用法和技巧十三:自动生成的TODO注释1

    使用eclipse的快捷键自动生成的代码,经常有这样的注释. 一眼看上去这个注释和一般的注释并无什么差别,不过TODO这个字符串的颜色不一样,应该有些内容.TODO是eclipse中提供的一种任务标签 ...

  9. kali之ARP欺骗获取图片流

    其实很简单,就两步: 1. 后接三个参数: 网卡:eth0    网关:10.0.0.1    攻击目标:10.0.0.128 2.启动监听 会弹出一个框 里面就会显示攻击目标通过浏览器访问的页面上的 ...

  10. axure制作圆形组件——axure制作技巧

    Axure本身是没有直接提供圆形组件的,所以很多朋友在微博上问,如何使用axure制作圆形,难道都要找美工-- Axure没有提供圆形组件,但是它提供了一个万能组件--矩形组件,只要有矩形组件,我们就 ...