luoguP2572 [SCOI2010]序列操作
题目&&链接
反正数据都是一样的,luogu比较友好
luogu
bzoj
lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作:
0 a b 把[a, b]区间内的所有数全变成0
1 a b 把[a, b]区间内的所有数全变成1
2 a b 把[a,b]区间内的所有数全部取反,也就是说把所有的0变成1,把所有的1变成0
3 a b 询问[a, b]区间内总共有多少个1
4 a b 询问[a, b]区间内最多有多少个连续的1
思路
每一步操作线段树都是可以维护的
但是,合起来也确实太恶心了点
维护 区间和,1的最长连续,前缀连续,后缀连续,
以及0的最长连续,前缀连续,后缀连续
7个值
以及两个lazy
一个是区间赋值的lazy1,一个是取反的lazy2
然后码代码就好了
注意*1
lazy下方的时候
只有三种情况
①先下方lazy1,后lazy2
②下方lazy1
③下方lazy2
因为如果是区间赋值的话,那么lazy2不lazy2的也没有啥子意义了
所以先lazy2,后lazy1这三种是被覆盖的
所以我们先下方lazy1
再下方lazy2
注意*2
还有,就是那个维护前缀后缀这种东西的时候
询问范围值不是int
因为你的答案不一定在线段树的一个儿子中,不能二分
所以要返回node_seg再进行选择
注意*3
写长代码的时候,像我一样没有一遍过变异的能力而且报错极多的
最后写一段编译一段,要不最后真的很绝望的
那些许多重复操作的
最好写点小函数之类的
减轻代码量
我只会输出调试,算了,不说了
/**************************************************************
Problem: 1858
User: 3010651817
Language: C++
Result: Accepted
Time:1496 ms
Memory:20436 kb
****************************************************************/
#include <iostream>
#include <cstdio>
#include <cstring>
#define ls rt<<1
#define rs rt<<1|1
using namespace std;
const int maxn = 1e5 + 7;
int n, m, a[maxn];
struct node {
int l, r, size;
int lazy1 , lazy2;
int sum;
int lk, rk, k;
int lk_, rk_, k_;
void add(int a) {
lk = rk = k = sum = a;
}
void add_(int a) {
lk_ = rk_ = k_ = a;
sum = size - a;
}
void swap_() {
swap(k_, k);
swap(lk_, lk);
swap(rk_, rk);
sum = size - sum;
}
} e[maxn << 2];
void pushdown_lazy1(int rt) {
e[ls].add(e[rt].lazy1 * e[ls].size);
e[ls].add_((1 ^ e[rt].lazy1)*e[ls].size);
e[ls].lazy1 = e[rt].lazy1;
e[ls].lazy2 = 0;
e[rs].add(e[rt].lazy1 * e[rs].size);
e[rs].add_((1 ^ e[rt].lazy1)*e[rs].size);
e[rs].lazy1 = e[rt].lazy1;
e[rs].lazy2 = 0;
}
void pushdown_lazy2(int rt) {
e[ls].swap_();
e[ls].lazy2 = !e[ls].lazy2;
e[rs].swap_();
e[rs].lazy2 = !e[rs].lazy2;
}
int read() {
int x = 0, f = 1; char s = getchar();
for (; s < '0' || s > '9'; s = getchar()) if (s == '-') f = -1;
for (; s >= '0' && s <= '9'; s = getchar()) x = x * 10 + s - '0';
return x * f;
}
int max_2(int a, int b, int c) {
return max(max(a, b), c);
}
void pushup(int rt) {
e[rt].sum = e[ls].sum + e[rs].sum;
e[rt].k = max_2(e[ls].k, e[rs].k, e[ls].rk + e[rs].lk);
e[rt].lk = (e[ls].lk == e[ls].size) ? e[ls].lk + e[rs].lk : e[ls].lk;
e[rt].rk = (e[rs].rk == e[rs].size) ? e[ls].rk + e[rs].rk : e[rs].rk;
e[rt].k_ = max_2(e[ls].k_, e[rs].k_, e[ls].rk_ + e[rs].lk_);
e[rt].lk_ = (e[ls].lk_ == e[ls].size) ? e[ls].lk_ + e[rs].lk_ : e[ls].lk_;
e[rt].rk_ = (e[rs].rk_ == e[rs].size) ? e[ls].rk_ + e[rs].rk_ : e[rs].rk_;
}
void pushdown(int rt) {
if (e[rt].lazy1 != -1)
pushdown_lazy1(rt);
if (e[rt].lazy2 == 1)
pushdown_lazy2(rt);
e[rt].lazy1 = -1;
e[rt].lazy2 = 0;
}
void build(int l, int r, int rt) {
e[rt].l = l, e[rt].r = r, e[rt].size = r - l + 1;
e[rt].lazy1 = -1, e[rt].lazy2 = 0;
if (l == r) {
e[rt].add(a[l]);
e[rt].add_(!a[l]);
return;
}
int mid = (l + r) >> 1;
build(l, mid, ls);
build(mid + 1, r, rs);
pushup(rt);
}
void update_1(int L, int R, int k, int rt) {
if (L <= e[rt].l && e[rt].r <= R) {
e[rt].add(k * e[rt].size);
e[rt].add_((1 ^ k)*e[rt].size);
e[rt].lazy1 = k;
e[rt].lazy2 = 0;
return;
}
pushdown(rt);
int mid = (e[rt].l + e[rt].r) >> 1;
if (L <= mid) update_1(L, R, k, ls);
if (R > mid) update_1(L, R, k, rs);
pushup(rt);
}
void update_2(int L, int R, int rt) {
if (L <= e[rt].l && e[rt].r <= R) {
e[rt].swap_();
e[rt].lazy2 = !e[rt].lazy2;
return;
}
pushdown(rt);
int mid = (e[rt].l + e[rt].r) >> 1;
if (L <= mid) update_2(L, R, ls);
if (R > mid) update_2(L, R, rs);
pushup(rt);
}
int query_1(int L, int R , int rt) {
if (L <= e[rt].l && e[rt].r <= R) {
return e[rt].sum;
}
pushdown(rt);
int mid = (e[rt].l + e[rt].r) >> 1, ans = 0;
if (L <= mid) ans += query_1(L, R, ls);
if (R > mid) ans += query_1(L, R, rs);
pushup(rt);
return ans;
}
node query_2(int L, int R, int rt) {
if (L <= e[rt].l && e[rt].r <= R) {
return e[rt];
}
pushdown(rt);
int mid = (e[rt].l + e[rt].r) >> 1;
if (L <= mid && R > mid) {
node a = query_2(L, R, ls) , b = query_2(L, R, rs);
node tmp = {};
tmp.k = max_2(a.k, b.k, a.rk + b.lk);
tmp.lk = (a.lk == a.size) ? a.lk + b.lk : a.lk;
tmp.rk = (b.rk == b.size) ? a.rk + b.rk : b.rk;
return tmp;
} else if (L <= mid) return query_2(L, R, ls);
else if (R > mid) return query_2(L, R, rs);
}
int main() {
n = read(), m = read();
for (int i = 1; i <= n; ++i) {
a[i] = read();
}
build(1, n, 1);
for (int i = 1; i <= m; ++ i) {
int tmp = read(), a = read(), b = read();
a++, b++;
if (tmp == 0) {
update_1(a, b, 0, 1);
} if (tmp == 1) {
update_1(a, b, 1, 1);
} else if (tmp == 2) {
update_2(a, b, 1);
} else if (tmp == 3) {
printf("%d\n", query_1(a, b, 1));
} else if (tmp == 4) {
printf("%d\n", query_2(a, b, 1).k);
}
}
return 0;
}
luoguP2572 [SCOI2010]序列操作的更多相关文章
- bzoj 1858: [Scoi2010]序列操作
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MB 线段树,对于每个区间需要分别维护左右和中间的1和0连续个数,并在op=4时特殊 ...
- BZOJ 1858: [Scoi2010]序列操作( 线段树 )
略恶心的线段树...不过只要弄清楚了AC应该不难.... ---------------------------------------------------------------- #inclu ...
- bzoj1858[Scoi2010]序列操作 线段树
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 3079 Solved: 1475[Submit][Statu ...
- BZOJ_1858_[Scoi2010]序列操作_线段树
BZOJ_1858_[Scoi2010]序列操作_线段树 Description lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询 ...
- 【题解】Luogu P2572 [SCOI2010]序列操作
原题传送门:P2572 [SCOI2010]序列操作 这题好弱智啊 裸的珂朵莉树 前置芝士:珂朵莉树 窝博客里对珂朵莉树的介绍 没什么好说的自己看看吧 操作1:把区间内所有数推平成0,珂朵莉树基本操作 ...
- P2572 [SCOI2010]序列操作
对自己 & \(RNG\) : 骄兵必败 \(lpl\)加油! P2572 [SCOI2010]序列操作 题目描述 lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要 ...
- BZOJ1858 [Scoi2010]序列操作(线段树)
题目链接 [Scoi2010]序列操作 考验代码能力的一道好题. 思想还是很简单的(直接上线段树),但是比较难写. #include <bits/stdc++.h> using names ...
- (WAWAWAWAWAWA) BZOJ 1858: [Scoi2010]序列操作
二次联通门 : BZOJ 1858: [Scoi2010]序列操作 /* BZOJ 1858: [Scoi2010]序列操作 已经... 没有什么好怕的的了... 16K的代码... 调个MMP啊.. ...
- 1858: [Scoi2010]序列操作
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MB Submit: 3397 Solved: 1624 [Submit][Statu ...
随机推荐
- LoadRunner-创建场景
录制完脚本,并调试运行正常后,想要模拟并发进行压力测试,需要创建场景. 1.点击Tools->Create Controller Scenario... 2.选择手工场景,并设置并发用户数.点击 ...
- python中执行shell命令行read结果
+++++++++++++++++++++++++++++ python执行shell命令1 os.system 可以返回运行shell命令状态,同时会在终端输出运行结果 例如 ipython中运行如 ...
- 新版微信小程序即将上线 新增微信支付功能
<经济参考报>消息,新版微信小程序正在测试中,有可能将在近期正式上线.新版小程序增加了“附近门店”功能的接口,微信用户可以通过定位功能,查看提供线下服务的各类门店,并直接利用小程序实现包括 ...
- OC导航栏跳转指定界面
方法一: [self.navigationController popToViewController:[self.navigationController.viewControllers objec ...
- 007-Redi-命令-脚本命令、链接命令、服务器命令、事务、HyperLogLog
Redis 脚本命令 下表列出了 redis 脚本常用命令: 序号 命令及描述 1 EVAL script numkeys key [key ...] arg [arg ...] 执行 Lua 脚本. ...
- (转)Kangle配置文件
kangle配置文件 (重定向自Kangle配置文件) 目录 [隐藏] 1配置文件介绍 2重新加载配置文件 3config 3.1request和response(配置访问控制) 3.2listen( ...
- [py][mx]django邮箱注册的验证码部分-django-simple-captcha库使用
邮箱注册-验证码 验证码使用第三方库django-simple-captcha 这个安装图形插件步骤官网有哦 - 1.Install django-simple-captcha via pip: pi ...
- 分布式文件系统ceph快速部署
架构图 配置ceph-deploy节点 管理节点配置ceph yum源 vim /etc/yum.repos.d/ceph.repo [ceph-noarch] name=Ceph noarch pa ...
- 全文搜索引擎Xapian
安装过程 安装xapian-core wget http://oligarchy.co.uk/xapian/1.0.13/xapian-core-1.0.13.tar.gztar zxvf xapia ...
- 174. Dungeon Game(动态规划)
The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...