BZOJ_1500_[NOI2005]维修数列_splay

题意:

分析:

节点维护从左开始的最大连续子段和,从右开始的最大连续子段和,区间的最大连续子段和

插入:重新建一棵树,把pos旋到根,把pos+1旋到根的右儿子,直接插到根的右儿子的左儿子上

删除:节点回收,用循环队列或者栈存一下删除的节点,新建的时候用。

修改和翻转正常做,这两个标记互不影响。记得翻转时左右儿子的信息都要交换。

代码:

#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 1000050
#define ls ch[p][0]
#define rs ch[p][1]
#define get(x) (x==ch[f[x]][1])
int ch[N][2],f[N],siz[N],val[N],n,m;
int cov[N],turn[N],maxl[N],maxr[N],maxm[N];
int S[N],top,sum[N],rt,a[N],sz,tot;
int other_root;
char opt[20];
void clear(int x){if(!x)return ;ch[x][0]=ch[x][1]=f[x]=val[x]=siz[x]=sum[x]=turn[x]=maxl[x]=maxr[x]=maxm[x]=0;cov[x]=0x3f3f3f3f;}
void print(int p)
{
if(!p)return ;
if(ls)print(ls);
printf("p=%d val[p]=%d\n",p,val[p]);
if(rs)print(rs);
}
int newnode(int v)
{
int p;
if(top)p=S[--top];
else p=++tot;
clear(p);
val[p]=v;siz[p]=1;
maxl[p]=maxr[p]=v>0?v:0;
maxm[p]=v;
return p;
}
void update(int p)
{
if(!p)return ;
siz[p]=siz[ls]+siz[rs]+1;
sum[p]=sum[ls]+sum[rs]+val[p];
maxl[p]=max(maxl[ls],sum[ls]+val[p]+maxl[rs]);
maxr[p]=max(maxr[rs],sum[rs]+val[p]+maxr[ls]);
maxm[p]=val[p];
maxm[p]=max(max(maxm[p],maxm[ls]),maxm[rs]);
maxm[p]=max(maxm[p],maxr[ls]+val[p]+maxl[rs]);
}
void pushdown(int p)
{
if(cov[p]!=0x3f3f3f3f)
{
int t=cov[p];
cov[p]=0x3f3f3f3f;
cov[ls]=cov[rs]=t;
val[ls]=val[rs]=t;
sum[ls]=siz[ls]*t;
sum[rs]=siz[rs]*t;
maxl[ls]=maxr[ls]=t>0?sum[ls]:0;
maxm[ls]=t>0?sum[ls]:t;
maxl[rs]=maxr[rs]=t>0?sum[rs]:0;
maxm[rs]=t>0?sum[rs]:t;
}
if(turn[p])
{
swap(ls,rs);
swap(maxl[ls],maxr[ls]);
swap(maxl[rs],maxr[rs]);
turn[ls]^=1;
turn[rs]^=1;
turn[p]=0;
}
}
void rotate(int x)
{
int y=f[x],z=f[y],k=get(x);
ch[y][k]=ch[x][k^1];f[ch[y][k]]=y;
ch[x][k^1]=y;f[y]=x;f[x]=z;
if(z) ch[z][ch[z][1]==y]=x;
update(y),update(x);
if(rt==y)rt=x;
}
void splay(int x,int y)
{
for(int fa;(fa=f[x])!=y;rotate(x))
if(f[fa]!=y)
rotate((get(x)==get(fa)) ? fa : x);
}
int find(int x)
{
int p=rt;
while(1)
{
pushdown(p);
if(x<=siz[ls])p=ls;
else
{
x-=siz[ls]+1;
if(!x) return p;
p=rs;
}
}
}
void print1()
{
for(int i=1;i<=sz;i++){int p=find(i);printf("p=%d val[p]=%d\n",p,val[p]);}
}
void build(int fa,int l,int r)
{
if(l>r)return ;
int mid=l+r>>1;
ch[fa][mid>fa]=mid;
f[mid]=fa;
val[mid]=a[mid-1];
siz[mid]=1;
build(mid,l,mid-1);
build(mid,mid+1,r);
update(mid);
}
void build_merge(int fa,int l,int r,int flg)
{
if(l>r)return ;
int mid=l+r>>1;
int p=newnode(a[mid]);
val[p]=a[mid];
ch[fa][flg]=p;
f[p]=fa;
build_merge(p,l,mid-1,0);
build_merge(p,mid+1,r,1);
update(p);
}
void insert(int x,int cnt)
{
int i;
for(i=1;i<=cnt;i++) scanf("%d",&a[i]);
sz+=cnt;
int p=x+1;
x=find(x);
p=find(p);
splay(x,f[rt]);
splay(p,rt);
build_merge(p,1,cnt,0);
}
void rec(int p)
{
if(!p)return ;
if(ls)rec(ls);
ls=0;
if(rs)rec(rs);
rs=0;
clear(p);
S[top++]=p;
}
void del(int x,int p)
{
x=find(x);
p=find(p);
splay(x,f[rt]);
splay(p,rt);
sz-=siz[ls];
rec(ls);
ls=0;
update(p);update(x);
}
void modify(int x,int p,int c)
{
x=find(x);
p=find(p);
splay(x,f[rt]);
splay(p,rt);
cov[ls]=c;
val[ls]=c;
sum[ls]=c*siz[ls];
maxl[ls]=maxr[ls]=c>0?sum[ls]:0;
maxm[ls]=c>0?sum[ls]:c;
update(p);update(x);
}
void reverse(int x,int p)
{
x=find(x);
p=find(p);
splay(x,f[rt]);
splay(p,rt);
turn[ls]^=1;
swap(maxl[ls],maxr[ls]);
update(p);update(x);
}
void getsum(int x,int p)
{
x=find(x);
p=find(p);
splay(x,f[rt]);
splay(p,rt);
printf("%d\n",sum[ls]);
}
void getmax_sum()
{
int x=1,p=sz;
x=find(x);
p=find(p);
splay(x,f[rt]);
splay(p,rt);
printf("%d\n",maxm[ls]);
}
int main()
{
scanf("%d%d",&n,&m);
memset(cov,0x3f,sizeof(cov));
sz=n+2;tot=n+2;
int i,x,y,z;
for(i=1;i<=n;i++) scanf("%d",&a[i]);
build(0,1,n+2);
rt=n+3>>1;
for(i=1;i<=m;i++)
{
scanf("%s",opt);
if(opt[2]=='S'){
scanf("%d%d",&x,&y);
insert(x+1,y);
}else if(opt[2]=='L'){
scanf("%d%d",&x,&y);
del(x,x+y+1);
}else if(opt[2]=='K'){
scanf("%d%d%d",&x,&y,&z);
modify(x,x+y+1,z);
}else if(opt[2]=='V'){
scanf("%d%d",&x,&y);
reverse(x,x+y+1);
}else if(opt[2]=='T'){
scanf("%d%d",&x,&y);
getsum(x,x+y+1);
}else{
getmax_sum();
}
}
}

BZOJ_1500_[NOI2005]维修数列_splay的更多相关文章

  1. [NOI2005] 维修数列

    1500: [NOI2005]维修数列 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 8397  Solved: 2530 Description In ...

  2. bzoj 1500: [NOI2005]维修数列 splay

    1500: [NOI2005]维修数列 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 6556  Solved: 1963[Submit][Status ...

  3. [BZOJ1500][NOI2005]维修数列---解题报告

    Portal Gun:[BZOJ1500][NOI2005]维修数列 有一段时间没写博客了,最近在刚数据结构......各种板子背得简直要起飞,题目也是一大堆做不完,这里就挑一道平衡树的题来写写好了 ...

  4. bzoj千题计划221:bzoj1500: [NOI2005]维修数列(fhq treap)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1500 1.覆盖标记用INF表示无覆盖标记,要求可能用0覆盖 2.代表空节点的0号节点和首尾的两个虚拟 ...

  5. [BZOJ1500][NOI2005]维修数列 解题报告 Splay

    Portal Gun:[BZOJ1500][NOI2005]维修数列 有一段时间没写博客了,最近在刚数据结构......各种板子背得简直要起飞,题目也是一大堆做不完,这里就挑一道平衡树的题来写写好了 ...

  6. BZOJ 1500: [NOI2005]维修数列 (splay tree)

    1500: [NOI2005]维修数列 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 4229  Solved: 1283[Submit][Status ...

  7. 【BZOJ1500】[NOI2005]维修数列 Splay

    [BZOJ1500][NOI2005]维修数列 Description Input 输入的第1 行包含两个数N 和M(M ≤20 000),N 表示初始时数列中数的个数,M表示要进行的操作数目.第2行 ...

  8. [bzoj1500][NOI2005]维修数列_非旋转Treap

    维修数列 bzoj-1500 NOI-2005 题目大意:给定n个数,m个操作,支持:在指定位置插入一段数:删除一个数:区间修改:区间翻转.查询:区间和:全局最大子序列. 注释:$1\le n_{ma ...

  9. 【BZOJ】1500: [NOI2005]维修数列

    [算法]splay [题解]数据结构 感谢Occult的模板>_<:HYSBZ 1500 维修数列 #include<cstdio> #include<cctype> ...

随机推荐

  1. JMM

    1.JMM简介 i.内存模型概述 Java平台自动集成了线程以及多处理器技术,这种集成程度比Java以前诞生的计算机语言要厉害很多,该语言针对多种异构平台的平台独立性而使用的多线程技术支持也是具有开拓 ...

  2. SpringBoot使用Maven插件打包部署

    [问题] 之前一直用SpringBoot做一些小项目,想打包部署在环境上,总是少依赖包jar.百度下可以通过Spring Boot Maven plugin插件,把Maven配置的依赖包都打到项目包里 ...

  3. 算法库中heap应用

    STL中make_heap 的接口为: default (1) template <class RandomAccessIterator> void make_heap (RandomAc ...

  4. 使用nginx缓存服务器上的静态文件

    一.nginx缓存的优点 如图所示,nginx缓存,可以在一定程度上,减少源服务器的处理请求压力. 因为静态文件(比如css,js, 图片)中,很多都是不经常更新的.nginx使用proxy_cach ...

  5. hadoop is running beyond virtual memory limits问题解决

    单机搭建了2.6.5的伪分布式集群,写了一个tf-idf计算程序,分词用的是结巴分词,使用standalone模式运行没有任何问题,切换到伪分布式模式运行一直报错: hadoop is running ...

  6. php中的session_id详解

    php中session_id()函数原型及说明session_id()函数说明:stringsession_id([string$id])session_id() 可以用来获取/设置 当前会话 ID. ...

  7. CentOS6系列系统启动常见故障排查与解决方法

    情景一.内核文件损坏 /boot/vmlinuz-2.6.32-642.el6.x86_64 内核文件 1.故障现象 2.解决方法:挂载光盘,进入rescue(救援)模式 3.选择--English- ...

  8. Version 1.6.0 of the JVM is not suitable for the this product.Version:1.8 or greater is required

    这个问题时在打开eclipse时报的一个错误,报这个问题的意思我们都明白,说的就是当前版本的jdk版本太低,eclipse需要更高版本的jdk. 那就下一个更高版本的jdk就可以啦,这里我要说说我当时 ...

  9. WBS 与 甘特图

    WBS:工作分解结构(Work Breakdown Structure) 创建WBS:创建WBS是把项目 交付成果和项目工作分解成较小的,更易于管理的组成部分的过程. WBS是项目管理重要的专业术语之 ...

  10. linux中~/cut/argus/

    1.Linux shell 截取字符变量的前8位 实现方法有如下几种: expr substr "$a" 1 8 echo $a|awk '{print substr(,1,8)} ...