BZOJ_1500_[NOI2005]维修数列_splay
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的更多相关文章
- [NOI2005] 维修数列
1500: [NOI2005]维修数列 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 8397 Solved: 2530 Description In ...
- bzoj 1500: [NOI2005]维修数列 splay
1500: [NOI2005]维修数列 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 6556 Solved: 1963[Submit][Status ...
- [BZOJ1500][NOI2005]维修数列---解题报告
Portal Gun:[BZOJ1500][NOI2005]维修数列 有一段时间没写博客了,最近在刚数据结构......各种板子背得简直要起飞,题目也是一大堆做不完,这里就挑一道平衡树的题来写写好了 ...
- bzoj千题计划221:bzoj1500: [NOI2005]维修数列(fhq treap)
http://www.lydsy.com/JudgeOnline/problem.php?id=1500 1.覆盖标记用INF表示无覆盖标记,要求可能用0覆盖 2.代表空节点的0号节点和首尾的两个虚拟 ...
- [BZOJ1500][NOI2005]维修数列 解题报告 Splay
Portal Gun:[BZOJ1500][NOI2005]维修数列 有一段时间没写博客了,最近在刚数据结构......各种板子背得简直要起飞,题目也是一大堆做不完,这里就挑一道平衡树的题来写写好了 ...
- BZOJ 1500: [NOI2005]维修数列 (splay tree)
1500: [NOI2005]维修数列 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 4229 Solved: 1283[Submit][Status ...
- 【BZOJ1500】[NOI2005]维修数列 Splay
[BZOJ1500][NOI2005]维修数列 Description Input 输入的第1 行包含两个数N 和M(M ≤20 000),N 表示初始时数列中数的个数,M表示要进行的操作数目.第2行 ...
- [bzoj1500][NOI2005]维修数列_非旋转Treap
维修数列 bzoj-1500 NOI-2005 题目大意:给定n个数,m个操作,支持:在指定位置插入一段数:删除一个数:区间修改:区间翻转.查询:区间和:全局最大子序列. 注释:$1\le n_{ma ...
- 【BZOJ】1500: [NOI2005]维修数列
[算法]splay [题解]数据结构 感谢Occult的模板>_<:HYSBZ 1500 维修数列 #include<cstdio> #include<cctype> ...
随机推荐
- eclipse调试的方法和技巧
eclipse调试图标所代表的含义: Step into 单步进入-将进入执行的方法内部继续执行. Step over 单步前进-执行下一步. Step return – 单步退出-跳出正在执行的方 ...
- Java多线程 阻塞队列和并发集合
转载:大关的博客 Java多线程 阻塞队列和并发集合 本章主要探讨在多线程程序中与集合相关的内容.在多线程程序中,如果使用普通集合往往会造成数据错误,甚至造成程序崩溃.Java为多线程专门提供了特有的 ...
- python中元组、列表、字典、集合知识
像列表一样处理字符串: 仅需要看字符串的首字符就知道如何处理该字符串的情况也很常见.例如,如果有一个姓与名的列表,您可以使用与列表相同的语法查看名与姓的第一个字符.这种看待字符串的方法叫做分片(sli ...
- redis 设置
设置成服务命令,redis目录下,执行cmd命令 redis-server --service-install redis.windows-service.conf --loglevel verbos ...
- asp.net core ABP模板本地化设置
ABP的语言本地化设置非常方便,甚至地区图标ABP框架都已经有了. 先看看结果吧. 英文的界面 中文的界面 配置流程如下: 首先在Localization目录下新建一个对应的json文件,里面存放对应 ...
- SpringCloud实战-Ribbon客户端负载均衡
前面我们已经完成了注册中心和服务提供者两个基础组件.接着介绍使用Spring Cloud Ribbon在客户端负载均衡的调用服务. ribbon 是一个客户端负载均衡器,可以简单的理解成类似于 ngi ...
- AUTOSAR分层-MCAL辨析
8. AUTOSAR中MCAL虽然包含各种drvier,但毕竟是AL即抽象层,不应包含architecture和device特定的信息.应该只包含模型定义,不包含实现细节. AUTOSAR文档中的 ...
- 洛谷 P2725 解题报告
P2725 邮票 Stamps 题目背景 给一组 N 枚邮票的面值集合(如,{1 分,3 分})和一个上限 K -- 表示信封上能够贴 K 张邮票.计算从 1 到 M 的最大连续可贴出的邮资. 题目描 ...
- 条件随机场CRF(三) 模型学习与维特比算法解码
条件随机场CRF(一)从随机场到线性链条件随机场 条件随机场CRF(二) 前向后向算法评估标记序列概率 条件随机场CRF(三) 模型学习与维特比算法解码 在CRF系列的前两篇,我们总结了CRF的模型基 ...
- 杨老师课堂_Java核心技术下之控制台模拟记事本案例
预览效果图: 背景介绍: 编写一个模拟记事本的程序通过在控制台输入指令,实现在本地新建文件打开文件和修改文件等功能. 要求在程序中: 用户输入指令1代表"新建文件",此时可以从控制 ...