2018.08.06 bzoj1500: [NOI2005]维修数列(非旋treap)
传送门
平衡树好题。
我仍然是用的fhqtreap,感觉速度还行。
维护也比线段树splay什么的写起来简单。
%%%非旋treap大法好。
代码:
#include<bits/stdc++.h>
#define N 500005
#define inf 0x3f3f3f3f
using namespace std;
queue<int>garbage;
typedef pair<int,int> res;
int n,m,rt=0,cnt=0,a[N],son[N][2],siz[N],rd[N],val[N],ls[N],rs[N],ms[N],sum[N],rev[N],cov[N];
inline int read(){
int ans=0,w=1;
char ch=getchar();
while(!isdigit(ch)){if(ch=='-')w=-1;ch=getchar();}
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans*w;
}
inline int max(int a,int b){return a>b?a:b;}
inline int min(int a,int b){return a<b?a:b;}
inline void swap(int&a,int&b){a^=b,b^=a,a^=b;}
inline int newnode(int v){
int x;
if(!garbage.empty())x=garbage.front(),garbage.pop();
else x=++cnt;
son[x][0]=son[x][1]=rev[x]=0,cov[x]=inf,siz[x]=1,rd[x]=rand(),val[x]=sum[x]=ms[x]=v,ls[x]=rs[x]=max(v,0);
return x;
}
inline void pushup(int p){
if(!son[p][0]&&!son[p][1]){siz[p]=1,sum[p]=ms[p]=val[p],ls[p]=rs[p]=max(val[p],0);return;}
siz[p]=siz[son[p][0]]+siz[son[p][1]]+1;
sum[p]=sum[son[p][0]]+sum[son[p][1]]+val[p];
if(son[p][0]&&son[p][1]){
ms[p]=max(max(ms[son[p][0]],ms[son[p][1]]),rs[son[p][0]]+val[p]+ls[son[p][1]]);
ls[p]=max(ls[son[p][0]],sum[son[p][0]]+val[p]+ls[son[p][1]]);
rs[p]=max(rs[son[p][1]],sum[son[p][1]]+val[p]+rs[son[p][0]]);
return;
}
if(son[p][0]){
ms[p]=max(ms[son[p][0]],rs[son[p][0]]+val[p]);
ls[p]=max(ls[son[p][0]],sum[son[p][0]]+val[p]);
rs[p]=max(0,val[p]+rs[son[p][0]]);
return;
}
ms[p]=max(ms[son[p][1]],ls[son[p][1]]+val[p]);
rs[p]=max(rs[son[p][1]],sum[son[p][1]]+val[p]);
ls[p]=max(0,val[p]+ls[son[p][1]]);
}
inline void pushrev(int p){swap(son[p][0],son[p][1]),swap(ls[p],rs[p]),rev[p]^=1;}
inline void pushcov(int p,int v){
sum[p]=v*siz[p],val[p]=cov[p]=v;
ls[p]=rs[p]=max(sum[p],0);
ms[p]=max(sum[p],v);
}
inline void pushdown(int p){
if(rev[p]){
if(son[p][0])pushrev(son[p][0]);
if(son[p][1])pushrev(son[p][1]);
rev[p]=0;
}
if(cov[p]!=inf){
if(son[p][0])pushcov(son[p][0],cov[p]);
if(son[p][1])pushcov(son[p][1],cov[p]);
cov[p]=inf;
}
}
inline int build(int dat[],int len){
int p,las=0;
static int stk[N],top;
for(int i=1;i<=len;++i){
p=newnode(dat[i]),las=0;
while(top&&rd[stk[top]]>rd[p])pushup(stk[top]),las=stk[top],stk[top--]=0;
if(top)son[stk[top]][1]=p;
son[p][0]=las,stk[++top]=p;
}
while(top)pushup(stk[top--]);
return stk[1];
}
inline int merge(int a,int b){
if(!a||!b)return a+b;
pushdown(a),pushdown(b);
if(rd[a]<rd[b]){son[a][1]=merge(son[a][1],b),pushup(a);return a;}
son[b][0]=merge(a,son[b][0]),pushup(b);return b;
}
inline res split(int p,int k){
if(!p)return res{0,0};
pushdown(p);
res ans,tmp;
if(siz[son[p][0]]>=k){
tmp=split(son[p][0],k);
ans.first=tmp.first,son[p][0]=tmp.second,pushup(p),ans.second=p;
return ans;
}
tmp=split(son[p][1],k-siz[son[p][0]]-1);
ans.second=tmp.second,son[p][1]=tmp.first,pushup(p),ans.first=p;
return ans;
}
inline void garbages(int p){
if(!p)return;
garbage.push(p),garbages(son[p][0]),garbages(son[p][1]);
}
inline void ins(){
int pos=read(),len=read();
static int dat[N];
for(int i=1;i<=len;++i)dat[i]=read();
int rtt=build(dat,len);
res x=split(rt,pos);
rt=merge(merge(x.first,rtt),x.second);
}
inline void del(){
int pos=read(),len=read();
res x=split(rt,pos-1),y=split(x.second,len);
rt=merge(x.first,y.second);
garbages(y.first);
}
inline void modify(){
int pos=read(),len=read(),v=read();
res x=split(rt,pos-1),y=split(x.second,len);
pushcov(y.first,v);
rt=merge(x.first,merge(y.first,y.second));
}
inline void reverse(){
int pos=read(),len=read();
res x=split(rt,pos-1),y=split(x.second,len);
pushrev(y.first);
rt=merge(x.first,merge(y.first,y.second));
}
inline void query(){
int pos=read(),len=read();
res x=split(rt,pos-1),y=split(x.second,len);
printf("%d\n",sum[y.first]);
rt=merge(x.first,merge(y.first,y.second));
}
int main(){
srand(time(NULL));
n=read(),m=read();
for(int i=1;i<=n;++i)a[i]=read();
rt=build(a,n);
while(m--){
char s[20];
scanf("%s",s);
if(s[0]=='I')ins();
else if(s[0]=='D')del();
else if(s[0]=='M'&&s[2]=='K')modify();
else if(s[0]=='R')reverse();
else if(s[0]=='G')query();
else printf("%d\n",ms[rt]);
}
return 0;
}
2018.08.06 bzoj1500: [NOI2005]维修数列(非旋treap)的更多相关文章
- BZOJ1500[NOI2005]维修数列——非旋转treap
题目描述 请写一个程序,要求维护一个数列,支持以下 6 种操作: 请注意,格式栏 中的下划线‘ _ ’表示实际输入文件中的空格 输入 输入的第1 行包含两个数N 和M(M ≤20 000),N 表示初 ...
- 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 ...
- 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]维修数列[splay ***]
1500: [NOI2005]维修数列 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 12278 Solved: 3880[Submit][Statu ...
- [bzoj1500][NOI2005]维修数列——splay
题目 题解 这道题可以说是数列问题的大BOSS,也算是这一周来学习splay等数据结构的一个总结. 我们一个一个地看这些操作. 对于操作1,我们首先建一棵子树,直接接上原树即可. 对于操作2,我们找到 ...
随机推荐
- MySQL命令行学习
1.登录mysql 本地:mysql -u root -p, 回车后输入密码; 也可以p后不加空格,直接加密码.回车就登录了 远程:mysql -hxx.xx.xx.xx -u -pxxx 2.查看数 ...
- J2SE 8的泛型
泛型的简单使用 1. 泛型一般用E表示集合中元素;k和v表示Map中的key和value;R表示return值;T/U/S表示任意类型 //(1) 简单单个元素的泛型 Box<String> ...
- php解析url并得到url中的参数
<?php $url = 'http://www.baidu.com/index.php?m=content&c=index&a=lists&catid=6&ar ...
- 机器学习入门-概率阈值的逻辑回归对准确度和召回率的影响 lr.predict_proba(获得预测样本的概率值)
1.lr.predict_proba(under_text_x) 获得的是正负的概率值 在sklearn逻辑回归的计算过程中,使用的是大于0.5的是正值,小于0.5的是负值,我们使用使用不同的概率结 ...
- 可视化库-seaborn-多变量分析绘图(第五天)
1. sns.stripplot(x='data', y='total_bill', data=tips, jitter=True), 画出竖形的样子,jitter=True为了使得数据尽量分开 im ...
- mysql创建定时器(event),查看定时器,打开定时器,设置定时器时间
由于项目需要创建定时器(evevt),所以就百度了一下,发现基本都是来源于一个模板,有些功能还不全,现在自己总结一下. 注:mysql版本是从5.1开始才支持event的.如果你的版本低于5.1就先升 ...
- log4j2搭建记录
今天新建了一个项目,自己弄的小玩意,想要做的正式点,就想引入日志.就想到了log4j2,经过几个小时的努力,还真的可以用了,下面就记录一下我是怎么做的. 下面是总的结构: 下面是MAVEN依赖: &l ...
- python-股票数据定向爬取
re.findall soup.find_all ---------Q---- for i in ***: ***可以是什么类型,主要是关心什么类型的不可以 ------------trackback ...
- https 证书传递、验证和数据加密、解密过程解析
我们都知道HTTPS能够加密信息,以免敏感信息被第三方获取.所以很多银行网站或电子邮箱等等安全级别较高的服务都会采用HTTPS协议. HTTPS简介 HTTPS其实是有两部分组成:HTTP + SSL ...
- IE6 PNG不透明问题 (只解决img标签的图片)
<script type='text/javascript' src="/script/ie6.pngfix.js"></script> 会让一些posit ...