HDU 3397 Sequence operation(区间合并 + 区间更新)
题目链接: pid=3397">http://acm.hdu.edu.cn/showproblem.php?pid=3397
题意:给定n个数,由0,1构成。共同拥有5种操作。
每一个操作输入3个数,op,a。b。
- op == 0。将区间[a,b]赋值为0。
- op == 1,将区间[a,b]赋值为1;
- op == 2。将区间[a。b]内的01反转;
- op == 3。查询区间[a。b]中1的个数。
- op == 4,查询区间[a。b]中连续1的最大长度。
思路:区间合并 + 区间更新。
每一个结点存7个数:
- 区间内1的个数c1。
- 从区间左端点開始连续1的最大长度l1。
- 以区间右端点结束连续1的最大长度r1。
- 区间内连续1的最大长度m1。
- 从区间左端点開始连续0的最大长度l0。
- 以区间右端点结束连续0的最大长度r0。
- 区间内连续0的最大长度m0。
记录0的连续是为了反转后统计结果方便。
代码:
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <math.h>
#include <algorithm>
#include <string>
using namespace std;
#define lson l, m, rt << 1
#define rson m + 1, r, rt << 1 | 1
const int N = 1e5 + 10;
const int INF = 0x7f7f7f7f;
struct Node {
int c1;
int l1, r1, m1;
int l0, r0, m0;
Node() {}
Node(int a, int b, int c, int d, int e, int f, int g) {
c1 = a;
l1 = b;
r1 = c;
m1 = d;
l0 = e;
r0 = f;
m0 = g;
}
void reversal(int s) {
c1 = s;
swap(l1, l0);
swap(r1, r0);
swap(m1, m0);
}
};
Node node[N << 2];
int cover[N << 2];
int XOR[N << 2];
void pushup(int rt, int l, int r) {
int m = (l + r) >> 1;
int lenl = m - l + 1;
int lenr = r - m;
node[rt].c1 = node[rt << 1].c1 + node[rt << 1 | 1].c1;
if (node[rt << 1].l1 == lenl)
node[rt].l1 = node[rt << 1].l1 + node[rt << 1 | 1].l1;
else
node[rt].l1 = node[rt << 1].l1;
if (node[rt << 1 | 1].r1 == lenr)
node[rt].r1 = node[rt << 1 | 1].r1 + node[rt << 1].r1;
else
node[rt].r1 = node[rt << 1 | 1].r1;
if (node[rt << 1].l0 == lenl)
node[rt].l0 = node[rt << 1].l0 + node[rt << 1 | 1].l0;
else
node[rt].l0 = node[rt << 1].l0;
if (node[rt << 1 | 1].r0 == lenr)
node[rt].r0 = node[rt << 1 | 1].r0 + node[rt << 1].r0;
else
node[rt].r0 = node[rt << 1 | 1].r0;
node[rt].m1 = max(node[rt << 1].r1 + node[rt << 1 | 1].l1,
max(node[rt << 1].m1, node[rt << 1 | 1].m1));
node[rt].m0 = max(node[rt << 1].r0 + node[rt << 1 | 1].l0,
max(node[rt << 1].m0, node[rt << 1 | 1].m0));
}
void pushdown(int rt, int l, int r) {
int m = (l + r) >> 1;
int lenl = m - l + 1;
int lenr = r - m;
if (cover[rt] != -1) {
if (cover[rt] == 1) {
node[rt << 1] = Node(lenl, lenl, lenl, lenl, 0, 0, 0);
node[rt << 1 | 1] = Node(lenr, lenr, lenr, lenr, 0, 0, 0);
}
else {
node[rt << 1] = Node(0, 0, 0, 0, lenl, lenl, lenl);
node[rt << 1 | 1] = Node(0, 0, 0, 0, lenr, lenr, lenr);
}
cover[rt << 1] = cover[rt << 1 | 1] = cover[rt];
XOR[rt << 1] = XOR[rt << 1 | 1] = 0;
cover[rt] = -1;
}
if (XOR[rt] != 0) {
node[rt << 1].reversal(lenl - node[rt << 1].c1);
node[rt << 1 | 1].reversal(lenr - node[rt << 1 | 1].c1);
if (cover[rt << 1] != -1)
cover[rt << 1] = 1 - cover[rt << 1];
else
XOR[rt << 1] = 1 - XOR[rt << 1];
if (cover[rt << 1 | 1] != -1)
cover[rt << 1 | 1] = 1 - cover[rt << 1 | 1];
else
XOR[rt << 1 | 1] = 1 - XOR[rt << 1 | 1];
XOR[rt] = 0;
}
}
void build(int l, int r, int rt) {
cover[rt] = -1;
XOR[rt] = 0;
if (l == r) {
int x;
scanf("%d", &x);
node[rt] = Node(x, x, x, x, 1 - x, 1 - x, 1 - x);
return ;
}
int m = (l + r) >> 1;
build(lson);
build(rson);
pushup(rt, l, r);
}
void update(int op, int L, int R, int l, int r, int rt) {
if (L <= l && r <= R) {
if (op == 2) {
node[rt].reversal(r - l + 1 - node[rt].c1);
if (cover[rt] != -1)
cover[rt] = 1 - cover[rt];
else
XOR[rt] = 1 - XOR[rt];
}
else {
node[rt] = (op == 0 ? Node(0, 0, 0, 0, r - l + 1, r - l + 1, r - l + 1) :
Node(r - l + 1, r - l + 1, r - l + 1, r - l + 1, 0, 0, 0));
cover[rt] = op;
XOR[rt] = 0;
}
return ;
}
pushdown(rt, l, r);
int m = (l + r) >> 1;
if (L <= m)
update(op, L, R, lson);
if (R > m)
update(op, L, R, rson);
pushup(rt, l, r);
}
Node query(int op, int L, int R, int l, int r, int rt) {
if (L <= l && r <= R) {
return node[rt];
}
pushdown(rt, l, r);
int m = (l + r) >> 1;
Node ql = Node(0, 0, 0, 0, 0, 0, 0);
Node qr = Node(0, 0, 0, 0, 0, 0, 0);
if (L <= m)
ql = query(op, L, R, lson);
if (R > m)
qr = query(op, L, R, rson);
pushup(rt, l, r);
if (op == 3) {
return Node(ql.c1 + qr.c1, 0, 0, 0, 0, 0, 0);
}
else {
Node res = Node(0, 0, 0, 0, 0, 0, 0);
res.m1 = max(ql.r1 + qr.l1, max(ql.m1, qr.m1));
if (ql.l1 == m - l + 1)
res.l1 = ql.l1 + qr.l1;
else
res.l1 = ql.l1;
if (qr.r1 == r - m)
res.r1 = qr.r1 + ql.r1;
else
res.r1 = qr.r1;
return res;
}
}
int main() {
int t_case;
scanf("%d", &t_case);
for (int i_case = 1; i_case <= t_case; i_case++) {
int n, m;
scanf("%d%d", &n, &m);
build(1, n, 1);
for (int i = 0; i < m; i++) {
int op, a, b;
scanf("%d%d%d", &op, &a, &b);
a++;
b++;
if (op <= 2) {
update(op, a, b, 1, n, 1);
}
else {
Node res = query(op, a, b, 1, n, 1);
if (op == 3)
printf("%d\n", res.c1);
else
printf("%d\n", res.m1);
}
}
}
return 0;
}
HDU 3397 Sequence operation(区间合并 + 区间更新)的更多相关文章
- hdu 3397 Sequence operation 线段树 区间更新 区间合并
题意: 5种操作,所有数字都为0或1 0 a b:将[a,b]置0 1 a b:将[a,b]置1 2 a b:[a,b]中的0和1互换 3 a b:查询[a,b]中的1的数量 4 a b:查询[a,b ...
- hdu 3397 Sequence operation (线段树 区间合并 多重标记)
链接:http://acm.hdu.edu.cn/showproblem.php?pid=3397 题意: 给你一串01串,有5种操作 0. 区间全部变为0 1.区间全部变为1 2.区间异或 3.询问 ...
- HDU 3397 Sequence operation(线段树)
HDU 3397 Sequence operation 题目链接 题意:给定一个01序列,有5种操作 0 a b [a.b]区间置为0 1 a b [a,b]区间置为1 2 a b [a,b]区间0变 ...
- 【线段树】HDU 3397 Sequence operation 区间合并
操作 Change operations: 0 a b change all characters into '0's in [a , b] 1 a b change all characters i ...
- hdu 3397 Sequence operation(很有意思的线段树题)
Sequence operation Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- HDU 3397 Sequence operation
题目:下列操作 Change operations:0 a b change all characters into '0's in [a , b]1 a b change all character ...
- hdu 3397 Sequence operation(线段树:区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3397 题意:给你一个长度为n的0,1序列,支持下列五种操作, 操作0(0 a b):将a到b这个区间的 ...
- HDU 3397 Sequence operation (区间合并,操作比较多)
费了我一天半的时间,到处debug,后来才发现,主要是建树的时候只在叶子节点对lazy1和lazy2进行初始化了,父节点都没初始化...晕. 具体见代码吧. #include <iostream ...
- (简单) HDU 3397 Sequence operation,线段树+区间合并。
Problem Description lxhgww got a sequence contains n characters which are all '0's or '1's. We have ...
随机推荐
- shell中的cut命令
转:http://blog.sina.com.cn/s/blog_5e77c61f0100hqky.html cut是以每一行为一个处理对象的,这种机制和sed是一样的.(关于sed的入门文章将在近期 ...
- 什么是HotSpot VM & 深入理解Java虚拟机 JVM
参考 http://book.2cto.com/201306/25434.html 另外,这篇文章也是从一个系列中得出的: <深入理解Java虚拟机:JVM高级特性与最佳实践(第2版)> ...
- 分布式服务自增长唯一ID小结
1.常用生成唯一ID的方式,例如UUID 2.生成唯一自自增长ID方式: 例如: Zookeeper的增加ID: redis的incr方法 mongodb的objectId 3.采用雪花模型 如下代码 ...
- mysql 存储过程 演示样例代码
drop procedure if exists P_SEQUENCE; /** 暂省略包 @AUTO LIANGRUI 2014/6/27 T_PRO_PRODUCT 表 排序 对整个表进行按序号排 ...
- angular 中表单验证的探索
需求 之前有一段时间做一个搜索查询 但是有很多限制条件,如果校验不成功需要给用户提示错误,当然项目用的是组件库的校验 我能否自己写一个? 其实 我是不会的!!! 探索 angular 的校验功能很强大 ...
- JAVA Eclipse如何修改Android程序名称
Values中修改strings.xml中的app_name即可 注意他是连接到AndroidManifest.xml文件的
- vector list map 遍历删除指定元素
#include <stdio.h> #include <stdint.h> #include <vector> #include <list> #in ...
- mysql常用命令和函数
一.DROP IF EXISTS DROP FUNCTION IF EXISTS fun;DROP TABLE IF EXISTS table; 二.数据表1.建立表CREATE TABLE test ...
- 关于其它模块的设计,有非常多须要自己去构建和完毕,在这里就简单地举几个样例来看看其它模块的设计。我们要做的就是有更改password模块,客户选择模块和关于本软件模块。更改password模块用来更改管理员的password,客户选择对话框模块用来选择已加入的客户,关于本软件模块用来说明客户管理系统的一些必要信息和制作人的信息。
五,其它模块设计 关于其它模块的设计,有非常多须要自己去构建和完毕,在这里就简单地举几个样例来看看其它模块的设计. 我们要做的就是有更改password模块.客户选择模 ...
- sql2000实现row_number
一.以PersonID,classid,dt_ClassData为条件进行分组,每个小组加序号,row_number在sql2005中不可用 方法一.sql2000及以上版本--以PersonID,c ...