[NOI2005]维护数列(luogu)

打这玩意儿真是要了我的老命

Description

请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际输入文件中的空格)

Code

#include <cstdio>
#include <cstdlib>
#include <algorithm>
#define TAGNONE 10000001
using namespace std;
const int N=1e6+;
struct node
{
int size,rev,ch[],res,prer,prel,tag,fa,sum,val;
void clear()
{
size=rev=ch[]=ch[]=res=prer=prel=fa=sum=val=;
tag=TAGNONE;
}
}f[N];
int x,id[N],tot,n,ru[N],rt,pos,len,d[N],m,cnt;
char s[];
int get()
{
if(tot==) return ++cnt;
int x=ru[tot--];
return x;
}
void c_val(int x,int y)
{
if(!x) return ;
f[x].tag=f[x].val=y;
f[x].sum=y*f[x].size;
f[x].prel=f[x].prer=max(,f[x].sum);
f[x].res=max(y,f[x].sum);
}
void c_rev(int x)
{
swap(f[x].ch[],f[x].ch[]);
swap(f[x].prer,f[x].prel);
f[x].rev^=;
}
void push_up(int x)
{
int lc=f[x].ch[],rc=f[x].ch[];
f[x].size=f[lc].size+f[rc].size+;
f[x].sum=f[lc].sum+f[rc].sum+f[x].val;
f[x].res=max(max(f[lc].res,f[rc].res),f[lc].prer+f[rc].prel+f[x].val);
f[x].prel=max(f[lc].prel,f[lc].sum+f[rc].prel+f[x].val);
f[x].prer=max(f[rc].prer,f[rc].sum+f[lc].prer+f[x].val);
}
void push_down(int x)
{
int lc=f[x].ch[],rc=f[x].ch[];
if(f[x].tag!=TAGNONE)
{
c_val(lc,f[x].tag);
c_val(rc,f[x].tag);
f[x].tag=TAGNONE;
}
if(f[x].rev)
{
c_rev(lc),c_rev(rc);
f[x].rev=;
}
}
int kth(int k)
{
int x=rt;
while()
{
push_down(x);
int lc=f[x].ch[],rc=f[x].ch[];
if(f[lc].size+==k) return x;
else if(f[lc].size>=k) x=lc;
else k-=f[lc].size+,x=rc;
}
}
void rotate(int x)
{
int y=f[x].fa,z=f[y].fa;
int wh=(x==f[y].ch[]);
f[f[x].ch[wh^]].fa=y;
f[y].ch[wh]=f[x].ch[wh^];
f[x].ch[wh^]=y;
f[y].fa=x,f[x].fa=z;
if(z) f[z].ch[y==f[z].ch[]]=x;
push_up(y),push_up(x);
}
void splay(int x,int y)
{
for(int fx=f[x].fa;fx=f[x].fa,fx!=y;rotate(x))
if(f[fx].fa!=y) rotate((fx==f[f[fx].fa].ch[])^(x==f[fx].ch[])?x:fx);
if(!y) rt=x;
}
int split(int pos,int len)
{
int x=kth(pos),y=kth(pos+len+);
splay(x,),splay(y,x);
return f[y].ch[];
}
void creat(int x,int y)
{
f[x].val=f[x].sum=f[x].res=y;
f[x].prel=f[x].prer=max(y,);
f[x].rev=,f[x].tag=TAGNONE;
f[x].size=;
}
void build(int l,int r,int fa)
{
int mid=(l+r)>>;
int x=id[mid],fx=id[fa];
if(l==r) creat(x,d[mid]);
if(l<mid) build(l,mid-,mid);
if(mid<r) build(mid+,r,mid);
f[x].tag=TAGNONE,f[x].val=d[mid],f[x].fa=fx;
push_up(x);
f[fx].ch[mid>=fa]=x;
}
void insert(int x,int len)
{
for(int i=;i<=len;i++)
scanf("%d",&d[i]),id[i]=get();
build(,len,);
int a=kth(x+),b=kth(x+);
splay(a,),splay(b,a);
int c=id[(+len)>>];
f[c].fa=b,f[b].ch[]=c;
push_up(b),push_up(a);
}
void remove(int x)
{
if(f[x].ch[]) remove(f[x].ch[]);
if(f[x].ch[]) remove(f[x].ch[]);
ru[++tot]=x,f[x].clear();
}
void eraser(int k,int len)
{
int x=split(k,len),y=f[x].fa;
remove(x);
f[y].ch[]=;
push_up(y),push_up(f[y].fa);
}
void update(int pos,int len,int k)
{
int x=split(pos,len);
c_val(x,k);
push_up(f[x].fa),push_up(rt);
}
void rever(int pos,int len)
{
int x=split(pos,len);
if(f[x].tag!=TAGNONE) return;
c_rev(x);
push_up(f[x].fa),push_up(rt);
}
void query(int pos,int len)
{
int x=split(pos,len);
printf("%d\n",f[x].sum);
}
int main()
{
scanf("%d%d",&n,&m);
f[].res=d[]=d[n+]=-<<;
for(int i=;i<=n;i++) scanf("%d",&d[i+]);
for(int i=;i<=n+;i++) id[i]=i;
build(,n+,);
rt=(n+)>>,cnt=n+;
while(m--)
{
scanf("%s",s);
if(s[]=='M' && s[]=='A' && s[]=='X') printf("%d\n",f[rt].res);
else scanf("%d%d",&pos,&len);
if(s[]=='I') insert(pos,len);
if(s[]=='D') eraser(pos,len);
if(s[]=='M' && s[]=='A' && s[]=='K')
scanf("%d",&x),update(pos,len,x);
if(s[]=='R') rever(pos,len);
if(s[]=='G') query(pos,len);
}
return ;
}

[NOI2005]维护数列(区间splay)的更多相关文章

  1. P2042 [NOI2005]维护数列 && Splay区间操作(四)

    到这里 \(A\) 了这题, \(Splay\) 就能算入好门了吧. 今天是个特殊的日子, \(NOI\) 出成绩, 大佬 \(Cu\) 不敢相信这一切这么快, 一下子机房就只剩我和 \(zrs\) ...

  2. 洛谷 P2042 [NOI2005]维护数列-Splay(插入 删除 修改 翻转 求和 最大的子序列)

    因为要讲座,随便写一下,等讲完有时间好好写一篇splay的博客. 先直接上题目然后贴代码,具体讲解都写代码里了. 参考的博客等的链接都贴代码里了,有空再好好写. P2042 [NOI2005]维护数列 ...

  3. P2042 [NOI2005]维护数列[splay或非旋treap·毒瘤题]

    P2042 [NOI2005]维护数列 数列区间和,最大子列和(必须不为空),支持翻转.修改值.插入删除. 练码力的题,很毒瘤.个人因为太菜了,对splay极其生疏,犯了大量错误,在此记录,望以后一定 ...

  4. 数据结构(Splay平衡树):COGS 339. [NOI2005] 维护数列

    339. [NOI2005] 维护数列 时间限制:3 s   内存限制:256 MB [问题描述] 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际 ...

  5. [NOI2005] 维护数列

    [NOI2005] 维护数列 题目 传送门 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际输入文件中的空格) 操作编号 输入文件中的格式 说明 1 ...

  6. Luogu P2042 [NOI2005]维护数列(平衡树)

    P2042 [NOI2005]维护数列 题意 题目描述 请写一个程序,要求维护一个数列,支持以下\(6\)种操作:(请注意,格式栏中的下划线'_'表示实际输入文件中的空格) 输入输出格式 输入格式: ...

  7. [转载]无旋treap:从单点到区间(例题 BZOJ1500&NOI2005 维护数列 )

    转自ZZH大佬,原文:http://www.cnblogs.com/LadyLex/p/7182631.html 1500: [NOI2005]维修数列 Time Limit: 10 Sec  Mem ...

  8. [您有新的未分配科技点] 无旋treap:从单点到区间(例题 BZOJ1500&NOI2005 维护数列 )

    1500: [NOI2005]维修数列 Time Limit: 10 Sec  Memory Limit: 64 MB Description Input 输入的第1 行包含两个数N 和M(M ≤20 ...

  9. 【bzoj1500】 noi2005—维护数列

    http://www.lydsy.com/JudgeOnline/problem.php?id=1500 (题目链接) 题意 要求维护数列,操作有区间删除,区间插入,区间反转,区间修改,区间求和,求最 ...

随机推荐

  1. java_学生成绩管理系统

    //信1805-2 20183670 王云鹏 package student; import java.util.Scanner; public class ScoreManagement { sta ...

  2. 微信小程序map地图的一些使用注意事项

    1.小程序组件map,在微信7.0.4以上(不包括7.0.4)层级问题官方已作更新,可在map上随意添加任何标签使用z-index即可:微信7.0.4版本以下map组件层级默认是最高的,只能使用官方提 ...

  3. Autofac总结

    Autofac 介绍|术语 控制反转:IOC和DI IOC 调用者不再创建(不自己new)被调用者的实例,而是交给容器去创建(AutoFac就充当这里的容器),这就是控制反转 控制反转中引入的第三方对 ...

  4. 1084 外观数列 (20 分)C语言

    外观数列是指具有以下特点的整数序列: d, d1, d111, d113, d11231, d112213111, ... 它从不等于 1 的数字 d 开始,序列的第 n+1 项是对第 n 项的描述. ...

  5. 结合docker发布后端项目(基于gradle包管理)的shell脚本

    结合docker发布后端项目(基于gradle包管理)的shell脚本 本教程依据个人理解并经过实际验证为正确,特此记录下来,权当笔记. 注:基于linux操作系统(敏感信息都进行了处理) 目前主流的 ...

  6. Java集合概述(上)

    Java集合概述(上) 前言 先说说,为什么要写这么一篇博客(我总是喜欢写原因).因为最近到年底了,正好又要准备面试,所以在做各方面的技术总结.而Java集合是Java非常重要的一部分,自己前前后后也 ...

  7. spring boot集成spring-boot-starter-mail邮件功能

    前情提要 以目前IT系统功能来看,邮件功能是非常重要的一个功能.例如:找回密码.邮箱验证,邮件动态码.忘记密码,邮件营销等,都需要用到邮件功能.结合当下最流行的spring boot微服务,推出了sp ...

  8. 【DPDK】【CPU usage】DPDK应用如何计算当前系统的压力

    [前言] 使用DPDK开发的朋友应该都了解使用dpdk的fwd线程的工作模式是polling模式,即100%轮询的方式去加速网络IO,这样我们在操作系统层面上来观察目标processer会发现usag ...

  9. selenium中的xpath用法,使用xpath定位元素

    xpath路径选择器定位元素 1. xpath: 使用路径表达式来定位xml或者html中文档中选取节点.在 XPath 中,有七种类型的节点:元素.属性.文本.命名空间.处理指令.注释以及文档节点( ...

  10. 三个实用的javascript小技巧

    从后向前获取数组元素 如果你想从后向前获取一个数组的元素,可以这样写: var newArray = [1, 2, 3, 4] console.log(newArray.slice(-1)) // [ ...