「NOI2005」维护数列

传送门

维护过程有点像线段树。

但我们知道线段树的节点并不是实际节点,而平衡树的节点是实际节点。

所以在向上合并信息时要加入根节点信息。

然后节点再删除后编号要回退(栈),不然会爆空间。

具体实现看代码就好了。

参考代码:

#include <algorithm>
#include <cstdlib>
#include <cstdio>
#define rg register
#define file(x) freopen(x".in", "r", stdin), freopen(x".out", "w", stdout)
using namespace std;
template < class T > inline void read(T& s) {
s = 0; int f = 0; char c = getchar();
while ('0' > c || c > '9') f |= c == '-', c = getchar();
while ('0' <= c && c <= '9') s = s * 10 + c - 48, c = getchar();
s = f ? -s : s;
} const int _ = 1e6 + 2; int tot, tmp[_];
int n, q, rt, siz[_], pri[_], ch[2][_];
int val[_], mx[_], L[_], R[_], sum[_], rev[_], tag[_], flag[_]; inline int Newnode(int v) {
int id = tmp[tot--];
siz[id] = 1, pri[id] = rand();
val[id] = mx[id] = sum[id] = v, L[id] = R[id] = max(0, v);
ch[0][id] = ch[1][id] = rev[id] = flag[id] = 0;
return id;
} inline void pushup(int p) {
siz[p] = siz[ch[0][p]] + siz[ch[1][p]] + 1;
sum[p] = sum[ch[0][p]] + sum[ch[1][p]] + val[p];
L[p] = max(0, max(L[ch[0][p]], sum[ch[0][p]] + val[p] + L[ch[1][p]]));
R[p] = max(0, max(R[ch[1][p]], sum[ch[1][p]] + val[p] + R[ch[0][p]]));
mx[p] = max(val[p], R[ch[0][p]] + val[p] + L[ch[1][p]]);
if (ch[0][p]) mx[p] = max(mx[p], mx[ch[0][p]]);
if (ch[1][p]) mx[p] = max(mx[p], mx[ch[1][p]]);
} inline void Rev(int p) {
rev[p] ^= 1, swap(L[p], R[p]), swap(ch[0][p], ch[1][p]);
} inline void Tag(int p, int v) {
flag[p] = 1, val[p] = tag[p] = v, sum[p] = siz[p] * v;
L[p] = R[p] = max(0, sum[p]), mx[p] = max(val[p], sum[p]);
} inline void pushdown(int p) {
if (rev[p]) {
if (ch[0][p]) Rev(ch[0][p]);
if (ch[1][p]) Rev(ch[1][p]);
rev[p] = 0;
}
if (flag[p]) {
if (ch[0][p]) Tag(ch[0][p], tag[p]);
if (ch[1][p]) Tag(ch[1][p], tag[p]);
flag[p] = 0;
}
} inline int merge(int x, int y) {
if (!x || !y) return x + y;
if (pri[x] > pri[y])
return pushdown(x), ch[1][x] = merge(ch[1][x], y), pushup(x), x;
else
return pushdown(y), ch[0][y] = merge(x, ch[0][y]), pushup(y), y;
} inline void split(int p, int k, int& x, int& y) {
if (p) pushdown(p);
if (!p) { x = y = 0; return ; }
if (siz[ch[0][p]] + 1 <= k)
return x = p, split(ch[1][p], k - siz[ch[0][p]] - 1, ch[1][x], y), pushup(p);
else
return y = p, split(ch[0][p], k, x, ch[0][y]), pushup(p);
} inline void erase(int p) {
if (!p) return ;
tmp[++tot] = p;
if (ch[0][p]) erase(ch[0][p]);
if (ch[1][p]) erase(ch[1][p]);
} int main() {
srand((unsigned long long) new char);
read(n), read(q);
for (rg int i = 1; i <= 500000; ++i) tmp[++tot] = i;
for (rg int v, i = 1; i <= n; ++i) read(v), rt = merge(rt, Newnode(v));
char s[15];
for (int pos, x, v, a, b, c; q--; ) {
scanf("%s", s);
if (s[0] == 'I') {
read(pos), read(x);
split(rt, pos, a, b);
while (x--) read(c), a = merge(a, Newnode(c));
rt = merge(a, b);
}
if (s[0] == 'D') {
read(pos), read(x);
split(rt, pos - 1, a, b);
split(b, x, b, c);
erase(b);
rt = merge(a, c);
}
if (s[0] == 'M' && s[2] == 'K') {
read(pos), read(x), read(v);
split(rt, pos - 1, a, b);
split(b, x, b, c);
Tag(b, v);
rt = merge(a, merge(b, c));
}
if (s[0] == 'R') {
read(pos), read(x);
split(rt, pos - 1, a, b);
split(b, x, b, c);
Rev(b);
rt = merge(a, merge(b, c));
}
if (s[0] == 'G') {
read(pos), read(x);
split(rt, pos - 1, a, b);
split(b, x, b, c);
printf("%d\n", sum[b]);
rt = merge(a, merge(b, c));
}
if (s[0] == 'M' && s[2] == 'X')
printf("%d\n", mx[rt]);
}
return 0;
}

「NOI2005」维护数列的更多相关文章

  1. 【NOI2005】维护数列

    https://daniu.luogu.org/problem/show?pid=2042 一道伸展树维护数列的很悲伤的题目,共要维护两个标记和两个数列信息,为了维护MAX-SUM还要维护从左端开始的 ...

  2. ☆ [ZJOI2006] 书架 「平衡树维护数列」

    题目类型:平衡树 传送门:>Here< 题意:要求维护一个数列,支持:将某个元素置顶或置底,交换某元素与其前驱或后继的位置,查询编号为\(S\)的元素的排名,查询排名第\(k\)的元素编号 ...

  3. 数据结构(Splay平衡树):COGS 339. [NOI2005] 维护数列

    339. [NOI2005] 维护数列 时间限制:3 s   内存限制:256 MB [问题描述] 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际 ...

  4. [NOI2005] 维护数列

    [NOI2005] 维护数列 题目 传送门 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际输入文件中的空格) 操作编号 输入文件中的格式 说明 1 ...

  5. P2042 [NOI2005]维护数列 && Splay区间操作(四)

    到这里 \(A\) 了这题, \(Splay\) 就能算入好门了吧. 今天是个特殊的日子, \(NOI\) 出成绩, 大佬 \(Cu\) 不敢相信这一切这么快, 一下子机房就只剩我和 \(zrs\) ...

  6. 洛谷 P2042 [NOI2005]维护数列-Splay(插入 删除 修改 翻转 求和 最大的子序列)

    因为要讲座,随便写一下,等讲完有时间好好写一篇splay的博客. 先直接上题目然后贴代码,具体讲解都写代码里了. 参考的博客等的链接都贴代码里了,有空再好好写. P2042 [NOI2005]维护数列 ...

  7. 「BZOJ1485」[HNOI2009] 有趣的数列 (卡特兰数列)

    「BZOJ1485」[HNOI2009] 有趣的数列   Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai ...

  8. P2042 [NOI2005]维护数列[splay或非旋treap·毒瘤题]

    P2042 [NOI2005]维护数列 数列区间和,最大子列和(必须不为空),支持翻转.修改值.插入删除. 练码力的题,很毒瘤.个人因为太菜了,对splay极其生疏,犯了大量错误,在此记录,望以后一定 ...

  9. Luogu P2042 [NOI2005]维护数列(平衡树)

    P2042 [NOI2005]维护数列 题意 题目描述 请写一个程序,要求维护一个数列,支持以下\(6\)种操作:(请注意,格式栏中的下划线'_'表示实际输入文件中的空格) 输入输出格式 输入格式: ...

随机推荐

  1. 如何在app.js 和其他页面中更改globalData的值

    它不能用this.setData方法更改值,该方法只能更改data:{}对象(而且在app.js中无法使用该方法),因此用app.globalData.isLogin = true;

  2. 如何重启Cisco LAP?

    在Cisco WLC上进入对应的AP,能够看到Reset这个AP,但是这里会有两种选择: 看到上述的描述,可能很多人都不敢轻易的操作: 1.Hardware Reset:Perform a hardw ...

  3. springboot多模块项目打war包

    一.父模块配置 1,指定pakaging:pom 2,指定编译的版本:如下图: <properties> <project.build.sourceEncoding>UTF-8 ...

  4. 《一篇文章读懂HTTPS及其背后的加密原理》阅读笔记

    HTTPS(Hypertext Transfer Protocol Secure,超文本传输安全协议),是以安全为目标的HTTP通道,简单讲是HTTP的安全版.这篇文章深入介绍了它的原理. 当我们适用 ...

  5. 【模板】凸包向内推进求不严格的半平面交——poj3384

    想不明白这题写严格的半平面交为什么会错 /* 凸包所有边向内推进r */ #include<iostream> #include<cstring> #include<cs ...

  6. 【代码总结】MYSQL数据库的常见操作

    ============================== MYSQL数据库的常见操作 ============================== 一.mysql的连接与关闭 -h:指定所连接的服 ...

  7. Python中的lambda函数介绍

    Lambda函数,即Lambda 表达式(lambda expression),是一个匿名函数(不存在函数名的函数),Lambda表达式基于数学中的λ演算得名,直接对应于其中的lambda抽象(lam ...

  8. POJ2516 Minimum Cost

    亲爱的,一个货物销售者,现在遇到了一个大问题,他需要你的帮助.在他的销售区域有 N 个店主(从 1 到 N)向他储存货物,Dearboy 有M 个供应点(从 1 到 M),每个供应点提供 K 种不同的 ...

  9. 九、Appium-python-UI自动化之通过text定位

    1.通过xpath定位text xpath路径为://android.widget.EditText[@text='请输入包含街道的完整地址'] 2.通过AndroidUIAutomator # 这个 ...

  10. 无数据库模式kong/kong-ingress-controller

    apiVersion: v1kind: Namespacemetadata:  name: kong---apiVersion: apiextensions.k8s.io/v1beta1kind: C ...