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. G1 GC技术解析

    介绍 G1 GC,全称Garbage-First Garbage Collector,通过-XX:+UseG1GC参数来启用.G1收集器是工作在堆内不同分区上的收集器,分区既可以是年轻代也可以是老年代 ...

  2. jdk1.7 tomcat-7安装

    由于软件下载地址经常有变动,所以不能直接wget,还是直接到网上点击下载 下载jdk http://www.oracle.com/technetwork/java/javase/downloads/j ...

  3. rcp perspective 添加9个视图均匀排列

    PageLayout布局方法 pageLayout.addView("NodePMTSStream" + ":1", IPageLayout.TOP, 0.5f ...

  4. 使用Owin的WebApi,并分离Controllers

    1.新建空白web项目 2.添加新建项=>OWIN Startup类(此时会自动下载owin,放在Packages里) 3.在Startup中输入 public void Configurati ...

  5. HBase最佳实践 - 集群规划

    本文由  网易云发布. 作者:范欣欣 本篇文章仅限本站分享,如需转载,请联系网易获取授权. HBase自身具有极好的扩展性,也因此,构建扩展集群是它的天生强项之一.在实际线上应用中很多业务都运行在一个 ...

  6. Pydev Console中文提示乱码的问题

    1. 像这样的规则内容请这样处理"\u305d\u3093\u306a\u306b"style unicode string : print str.decode("un ...

  7. Django1.6版本的PG数据库定义手动升级

    Django1.7以后添加了migration功能,数据库定义的升级完全实现自动化,之前是通过一个叫south的app来做的.这篇文章谈一下1.6下的手动更新升级. 1.table create和ta ...

  8. error LNK2038: 检测到“RuntimeLibrary”的不匹配项: 值“MTd_StaticDebug”不匹配值“MDd_DynamicDebug

    属性1. 在工程上右键->属性->c/c++->代码生成->运行库 四个选项及含义分别如下: 1.1 /MDd:MD_DynamicDebug,我理解是 "共享DLL ...

  9. DevOps之四 Jenkins的安装与配置

    CentOS 上 Jenkins 安装 一.添加yum repos,然后安装 sudo wget -O /etc/yum.repos.d/jenkins.repo http://pkg.jenkins ...

  10. Windows10 ubuntu子系统的启用即基础配置

    Windows 10 在一周年更新后,本身集成一个不带有图形界面的ubuntu 14.04系统了,大大方便了Linux开发,并且本身使用很方便,像我这种Windows死忠,只会在Linux下跑一下一定 ...