BZOJ1500[NOI2005]维修数列——非旋转treap
题目描述
输入
输入的第1 行包含两个数N 和M(M ≤20 000),N 表示初始时数列中数的个数,M表示要进行的操作数目。
第2行包含N个数字,描述初始时的数列。
以下M行,每行一条命令,格式参见问题描述中的表格。
任何时刻数列中最多含有500 000个数,数列中任何一个数字均在[-1 000, 1 000]内。
插入的数字总数不超过4 000 000个,输入文件大小不超过20MBytes。
输出
对于输入数据中的GET-SUM和MAX-SUM操作,向输出文件依次打印结果,每个答案(数字)占一行。
样例输入
2 -6 3 5 1 -5 -3 6 3
GET-SUM 5 4
MAX-SUM
INSERT 8 3 -5 7 2
DELETE 12 1
MAKE-SAME 3 3 2
REVERSE 3 6
GET-SUM 5 4
MAX-SUM
样例输出
10
1
10
提示
这道题的区间操作十分全,无论是用非旋转treap还是splay写,都会有更深入的理解。
讲一下每个操作的实现
1、将插入的数再建一棵treap,然后把原treap断裂,把新建treap合并进去。
2、把treap断裂成三部分,然后再把左右两部分合并。
3、也是把treap断裂成三部分,把中间部分打标记,标记在合并时下传。
4、和上面一样,也是断裂成三部分,中间打标记。
5、断裂成三部分,输出中间部分根节点的子树和。
6、直接输出根节点的最大连续子段和。
最后不要忘了资源回收(回收节点编号)
最后附上代码(有注释)
#include<queue>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
int num;
int pos;
int tot;//树的大小
int root;//根节点
int n,k,g;
int x,y,z;
int b,c,d;
char ch[5];
queue<int>q;//资源回收队列
int a[200010];//插入序列及初始序列
int m[500010];//区间最大子段和
int s[500010];//翻转标记
int f[500010];//修改标记
int r[500010];//随机数
int ls[500010];//左儿子
int rs[500010];//右儿子
int lm[500010];//子树区间中从左端点开始的最大连续子段和
int rm[500010];//子树区间中以右端点结束的最大连续子段和
int val[500010];//节点权值
int sum[500010];//子树区间和
int size[500010];//子树大小
int INF=2147483647;
int insert(int v)
{
int x;
if(!q.empty())
{
x=q.front();
q.pop();
}
else
{
x=++tot;
}
size[x]=1;
ls[x]=rs[x]=s[x]=0;
f[x]=INF;
r[x]=rand();
val[x]=sum[x]=v;
lm[x]=rm[x]=m[x]=v;
return x;
}
void updata(int x)
{
if(!x)
{
return ;
}
size[x]=size[ls[x]]+size[rs[x]]+1;
sum[x]=sum[ls[x]]+sum[rs[x]]+val[x];
m[x]=max(max(0,rm[ls[x]])+val[x]+max(0,lm[rs[x]]),max(m[ls[x]],m[rs[x]]));
lm[x]=max(lm[ls[x]],sum[ls[x]]+val[x]+max(0,lm[rs[x]]));
rm[x]=max(rm[rs[x]],sum[rs[x]]+val[x]+max(0,rm[ls[x]]));
}
void change(int x,int v)
{
val[x]=v;
sum[x]=size[x]*v;
lm[x]=rm[x]=max(0,sum[x]);
m[x]=max(val[x],sum[x]);
f[x]=v;
}
void rotate(int x)
{
swap(ls[x],rs[x]);
swap(lm[x],rm[x]);
s[x]^=1;
}
void downdata(int x)
{
if(s[x])
{
if(ls[x])
{
rotate(ls[x]);
}
if(rs[x])
{
rotate(rs[x]);
}
}
if(f[x]!=INF)
{
if(ls[x])
{
change(ls[x],f[x]);
}
if(rs[x])
{
change(rs[x],f[x]);
}
}
s[x]=0;
f[x]=INF;
}
int build(int l,int r)
{
if(l>r)
{
return 0;
}
int mid=(l+r)>>1;
int v=a[mid];
int x=insert(v);
ls[x]=build(l,mid-1);
rs[x]=build(mid+1,r);
updata(x);
return x;
}
void split(int now,int t,int &x,int &y)
{
if(!now)
{
x=y=0;
}
else
{
downdata(now);
if(t<=size[ls[now]])
{
y=now;
split(ls[now],t,x,ls[now]);
}
else
{
x=now;
split(rs[now],t-size[ls[now]]-1,rs[now],y);
}
updata(now);
}
}
int merge(int x,int y)
{
if(!x||!y)
{
return x+y;
}
downdata(x);
downdata(y);
if(r[x]<r[y])
{
rs[x]=merge(rs[x],y);
updata(x);
return x;
}
else
{
ls[y]=merge(x,ls[y]);
updata(y);
return y;
}
}
void inque(int x)
{
if(!x)
{
return ;
}
q.push(x);
inque(ls[x]);
inque(rs[x]);
}
int main()
{
srand(16);
scanf("%d%d",&n,&k);
for(int i=1;i<=n;i++)
{
scanf("%d",&a[i]);
}
val[0]=m[0]=-INF;
root=build(1,n);
while(k--)
{
scanf("%s",ch);
if(ch[0]=='I')
{
scanf("%d%d",&pos,&num);
for(int i=1;i<=num;i++)
{
scanf("%d",&a[i]);
}
z=build(1,num);
split(root,pos,x,y);
root=merge(merge(x,z),y);
}
else if(ch[0]=='D')
{
scanf("%d%d",&pos,&num);
split(root,pos-1,x,y);
split(y,num,b,c);
root=merge(x,c);
inque(b);
}
else if(ch[2]=='K')
{
scanf("%d%d%d",&pos,&num,&g);
split(root,pos-1,x,y);
split(y,num,b,c);
change(b,g);
root=merge(x,merge(b,c));
}
else if(ch[0]=='R')
{
scanf("%d%d",&pos,&num);
split(root,pos-1,x,y);
split(y,num,b,c);
rotate(b);
root=merge(x,merge(b,c));
}
else if(ch[0]=='G')
{
scanf("%d%d",&pos,&num);
split(root,pos-1,x,y);
split(y,num,b,c);
printf("%d\n",sum[b]);
root=merge(x,merge(b,c));
}
else
{
printf("%d\n",m[root]);
}
}
return 0;
}
BZOJ1500[NOI2005]维修数列——非旋转treap的更多相关文章
- bzoj千题计划221:bzoj1500: [NOI2005]维修数列(fhq treap)
http://www.lydsy.com/JudgeOnline/problem.php?id=1500 1.覆盖标记用INF表示无覆盖标记,要求可能用0覆盖 2.代表空节点的0号节点和首尾的两个虚拟 ...
- [BZOJ1500][NOI2005]维修数列---解题报告
Portal Gun:[BZOJ1500][NOI2005]维修数列 有一段时间没写博客了,最近在刚数据结构......各种板子背得简直要起飞,题目也是一大堆做不完,这里就挑一道平衡树的题来写写好了 ...
- [BZOJ1500][NOI2005]维修数列 解题报告 Splay
Portal Gun:[BZOJ1500][NOI2005]维修数列 有一段时间没写博客了,最近在刚数据结构......各种板子背得简直要起飞,题目也是一大堆做不完,这里就挑一道平衡树的题来写写好了 ...
- [bzoj1500][NOI2005]维修数列_非旋转Treap
维修数列 bzoj-1500 NOI-2005 题目大意:给定n个数,m个操作,支持:在指定位置插入一段数:删除一个数:区间修改:区间翻转.查询:区间和:全局最大子序列. 注释:$1\le n_{ma ...
- 2018.08.06 bzoj1500: [NOI2005]维修数列(非旋treap)
传送门 平衡树好题. 我仍然是用的fhqtreap,感觉速度还行. 维护也比线段树splay什么的写起来简单. %%%非旋treap大法好. 代码: #include<bits/stdc++.h ...
- BZOJ1500[NOI2005]维修数列
Description Input 输入的第1 行包含两个数N 和M(M ≤20 000),N 表示初始时数列中数的个数,M表示要进行的操作数目.第2行包含N个数字,描述初始时的数列.以下M行,每行一 ...
- splay模板三合一 luogu2042 [NOI2005]维护数列/bzoj1500 [NOI2005]维修数列 | poj3580 SuperMemo | luogu3391 【模板】文艺平衡树(Splay)
先是维修数列 题解看这里,但是我写的跑得很慢 #include <iostream> #include <cstdio> using namespace std; int n, ...
- [bzoj1500][NOI2005]维修数列[Treap][可持久化Treap]
非旋转式Treap1500 :) #include <bits/stdc++.h> #pragma GCC optimize(3) using namespace std; const i ...
- BZOJ1500 [NOI2005]维修数列-fhq_Treap
题面见这里 反正是道平衡树,就拿 fhq_Treap 写了写... 这道题思路基本是围绕“用 Treap 维护中序遍历” 和 中序遍历的性质 来进行的操作 所以就可以类比线段树进行一些操作 1. 建树 ...
随机推荐
- 【Codeforces 142C】Help Caretaker
Codeforces 142 C 题意:给一个\(n\times m\)的空矩阵,求里面放最多的可旋转的\(T\)字形的个数,并输出方案. 思路1: 由于\(n\)和\(m\)比较小,所以可以尝试搜索 ...
- Recurrent Neural Network[CTC]
0. 背景 1. CTC原理 图 CTC结构图 CTC是看似和HMM有些联系,然后也采用DP来进行求解,将CTC结构图中<RNN输出,CTC层>单独拿出来,得到如下形式: 图 用前向-后向 ...
- Oracle ORA-01940: 无法删除当前连接的用户
当我们要删除一个oracle的用户时,如果有其他人连接到数据库则会报以下错误: ORA-01940: 无法删除当前连接的用户 处理办法就是:将连接到当前用户的session给kill掉. 处理步骤如下 ...
- [04] Bean的实例化和多个配置文件
之前我们已经提到过,Spring的核心能力之一就是IOC机制,从上章的示例中我们也可以看到,Spring中的ApplicationContext充当了一个实例化对象的容器的角色,并管理着它们的生命周期 ...
- click事件和mousedown、mouseup事件
点击select标签元素的时候,会弹出下拉.然而当option中没有元素时,就不希望弹出下拉(比如在某些浏览器中,点击select会默认出一个罩层效果,而此时没有数据选择的话,弹出比较不友好). 首先 ...
- 转:判断js中的数据类型的几种方法
判断js中的数据类型有一下几种方法:typeof.instanceof. constructor. prototype. $.type()/jquery.type(),接下来主要比较一下这几种方法的异 ...
- 将 C# 枚举序列化为 JSON 字符串 实践
一.定义枚举 public enum SiteTypeEnum { 中转部 = 1, 网点 = 2 } 还有 BooleanEnum 和 OptTypeEnum 这两个枚举,这里暂且省略了它们的定义. ...
- 【nodejs】让nodejs像后端mvc框架(asp.net mvc)一orm篇【如EF般丝滑】typeorm介绍(8/8)
文章目录 前情概要 在使用nodejs开发过程中,刚好碰到需要做一个小工具,需要用到数据库存储功能.而我又比较懒,一个小功能不想搞一个nodejs项目,又搞一个后端项目.不如直接在nodejs里面把对 ...
- Tomcat通过自带的Cluster方式实现Session会话共享环境操作记录
一般来说,在多个tomcat集群业务中,session会话共享是必须的需求,不然前端nginx转发过来的请求不知道之前请求在哪台tomcat节点上,从而就找不到session以至于最终导致请求失败.要 ...
- NEWBE CRALWER 产品需求文档
1.产品概述 本产品是学霸软件系统的爬虫部分,由NEWBE团队负责.主要任务是从网上爬取出相关数据后提供给C705组使用. 2.产品的发展经历 2.1 产品的发展经历 本产品从2014.10.29开始 ...