题目描述

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

输入输出格式

输入格式:

输入文件的第 1 行包含两个数 N 和 M,N 表示初始时数列中数的个数,M 表示要进行的操作数目。 第 2 行包含 N 个数字,描述初始时的数列。 以下 M 行,每行一条命令,格式参见问题描述中的表格

输出格式:

对于输入数据中的 GET-SUM 和 MAX-SUM 操作,向输出文件依次打印结 果,每个答案(数字)占一行。

输入输出样例

输入样例#1:

9 8

2 -6 3 5 1 -5 -3 6 3

GET-SUM 5 4

MAX-SUM

INSERT 8 3 -5 7 2

DELETE 12 1

MAKE-SAME 3 3 2

REVERSE 3 6

GET-SUM 5 4

MAX-SUM

输出样例#1:

-1

10

1

10

说明

你可以认为在任何时刻,数列中至少有 1 个数。

输入数据一定是正确的,即指定位置的数在数列中一定存在。

50%的数据中,任何时刻数列中最多含有 30 000 个数;

100%的数据中,任何时刻数列中最多含有 500 000 个数。

100%的数据中,任何时刻数列中任何一个数字均在[-1 000, 1 000]内。

100%的数据中,M ≤20 000,插入的数字总数不超过 4 000 000 。

分析

复杂度 思维1 编程7

本题思维上很简单,先判断使用平衡树,再用ls,rs维护最大和,之后挨个完成各个功能。注意节点需要回收利用。回收总复杂度O(n)。总体复杂度O(nlogn)。

代码

#include <cstdio>
#include <cstdlib>
#define rt ch[0][0]
const int N=500000,S=N+100,inf=(1<<25)-1;
int n,m,a[S],bin[S],ch[S][2],ls[S],rs[S],fa[S],su[S],sz[S],mx[S],add[S],rev[S];
void rd(int &x)
{
x=0;char c=getchar(),t=1;
while (c!='-' && (c<'0' || c>'9')) c=getchar();
if (c=='-') t=-1,c=getchar();
while (c>='0' && c<='9') x=(x<<3)+(x<<1)+(c-'0'),c=getchar();
x*=t;
}
void init()
{
for (int i=0;i<=N;i++)
bin[i]=i;
bin[0]=1; rt=1; ls[0]=rs[0]=mx[0]=-inf;
sz[0]=0; bin[0]=2;//use 1
sz[1]=2;ch[1][1]=2;fa[1]=0;a[1]=-inf;add[1]=inf; bin[0]=3;//use 2
sz[2]=1;fa[2]=1;add[2]=inf;a[2]=-inf;
}
inline int ma(int a,int b){if (a>b) return a;return b;}
inline int ma3(int a,int b,int c){return ma(a,ma(b,c));}
inline void swap(int &a,int &b){a^=b^=a^=b;}
inline void up(int x)
{
int L=ch[x][0],R=ch[x][1];
sz[x]=sz[L]+sz[R]+1;
su[x]=su[L]+su[R]+a[x];
ls[x]=ma3(ls[L],su[L]+a[x],su[L]+a[x]+ls[R]);
rs[x]=ma3(rs[R],su[R]+a[x],su[R]+a[x]+rs[L]);
int mid=ma(rs[L],0)+a[x]+ma(ls[R],0);
mx[x]=ma3(mx[L],mx[R],mid);
mx[x]=ma3(mx[x],ls[x],rs[x]);
}
inline void down(int x)
{
int &L=ch[x][0],&R=ch[x][1];
if (rev[x])
{
if (L)
{
rev[L]^=1;
swap(ch[L][0],ch[L][1]);
swap(ls[L],rs[L]);
}
if (R)
{
rev[R]^=1;
swap(ch[R][0],ch[R][1]);
swap(ls[R],rs[R]);
}
rev[x]=0;
}
if (add[x]!=inf)
{
if (L)
{
add[L]=add[x];
su[L]=add[L]*sz[L];
a[L]=add[L];
if (add[x]>0)
ls[L]=rs[L]=mx[L]=su[L];
else
ls[L]=rs[L]=mx[L]=a[L];
}
if (R)
{
add[R]=add[x];
su[R]=add[R]*sz[R];
a[R]=add[R];
if (add[x]>0)
ls[R]=rs[R]=mx[R]=su[R];
else
ls[R]=rs[R]=mx[R]=a[R];
}
add[x]=inf;
}
}
inline bool lor(int x){return ch[fa[x]][1]==x;}
inline void link(int x,int fat,int o){fa[x]=fat;ch[fat][o]=x;}
inline void rotate(int x)
{
int y=fa[x],r=fa[y];
down(y);down(x);
int rson=lor(y),yson=lor(x);
link(ch[x][yson^1],y,yson);
link(y,x,yson^1);
link(x,r,rson);
up(y);up(x);
}
void splay(int x,int to)
{
to=fa[to];
while (fa[x]!=to)
{
if (fa[fa[x]]==to) rotate(x);
else if (lor(x)==lor(fa[x])) rotate(fa[x]),rotate(x);
else rotate(x),rotate(x);
}
}
void build(int &k,int fat,int l,int r)
{
if (l>r)
{
k=0;
return;
}
k=bin[bin[0]++];
fa[k]=fat;
int mid=(l+r)>>1;
build(ch[k][0],k,l,mid-1);
rd(a[k]);
add[k]=inf;rev[k]=0;
build(ch[k][1],k,mid+1,r);
up(k);
}
void recycle(int k)
{
if (!k) return;
recycle(ch[k][0]);
a[k]=su[k]=ls[k]=rs[k]=mx[k]=fa[k]=add[k]=rev[k]=0;
bin[--bin[0]]=k;
recycle(ch[k][1]);
ch[k][0]=ch[k][1]=0;
}
int find(int x)
{
int o=rt;
while (o)
{
down(o);
if (sz[ch[o][0]]+1==x)
break;
if (x<=sz[ch[o][0]])
o=ch[o][0];
else x-=sz[ch[o][0]]+1,o=ch[o][1];
}
return o;
}
void insert(int pos,int tot)
{
pos++;
int bg=find(pos),ed=find(pos+1);
splay(bg,rt);
splay(ed,ch[bg][1]);
build(ch[ed][0],ed,1,tot);
up(ed);up(bg);
}
void del(int pos,int tot)
{
int bg=find(pos),ed=find(pos+tot+1);
splay(bg,rt);splay(ed,ch[bg][1]);
recycle(ch[ed][0]);
ch[ed][0]=0;
up(ed);up(bg);
}
void make_same(int pos,int tot,int x)
{
int bg=find(pos),ed=find(pos+tot+1);
splay(bg,rt);splay(ed,ch[bg][1]);
int o=ch[ed][0];
add[o]=x;su[o]=x*sz[o];a[o]=x;
if (x>0) ls[o]=rs[o]=mx[o]=su[o];
else ls[o]=rs[o]=mx[o]=a[o];
up(ed);up(bg);
}
void reverse(int pos,int tot)
{
int bg=find(pos),ed=find(pos+tot+1);
splay(bg,rt);splay(ed,ch[bg][1]);
down(bg);down(ed);
int o=ch[ed][0];
rev[o]=1;
swap(ch[o][0],ch[o][1]);
swap(ls[o],rs[o]);
up(ed);up(bg);
}
void getsum(int pos,int tot)
{
int bg=find(pos),ed=find(pos+tot+1);
splay(bg,rt);
splay(ed,ch[bg][1]);
int o=ch[ed][0];
printf("%d\n",su[o]);
}
int main()
{
rd(n);rd(m);
init();
build(ch[2][0],2,1,n);up(2);up(1);
char op[10];int tot,pos,x;
while (m--)
{
scanf("%s",op);
if (op[2]=='X')//max_sum
printf("%d\n",mx[rt]);
else
{
rd(pos);rd(tot);
if (op[2]=='S')//Insert
insert(pos,tot);
else if (op[2]=='L')
del(pos,tot);
else if ('K'==op[2])
{
scanf("%d",&x);
make_same(pos,tot,x);
}
else if ('V'==op[2])
reverse(pos,tot);
else getsum(pos,tot);
}
}
return 0;
}

Luogu P2042 [NOI2005]维护数列的更多相关文章

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

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

  2. BZOJ 1500 Luogu P2042 [NOI2005] 维护数列 (Splay)

    手动博客搬家: 本文发表于20180825 00:34:49, 原地址https://blog.csdn.net/suncongbo/article/details/82027387 题目链接: (l ...

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

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

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

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

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

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

  6. 洛谷P2042 [NOI2005]维护数列

    #include<cstdio> #include<cstdlib> #include<algorithm> #include<cstring> #in ...

  7. P2042 [NOI2005]维护数列

    思路 超级恶心的pushdown 昏天黑地的调 让我想起了我那前几个月的线段树2 错误 这恶心的一道题终于过了 太多错误,简直说不过来 pushup pushdown 主要就是这俩不太清晰,乱push ...

  8. [NOI2005]维护数列(区间splay)

    [NOI2005]维护数列(luogu) 打这玩意儿真是要了我的老命 Description 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际输入文 ...

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

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

随机推荐

  1. 怎么让小白理解intel处理器(CPU)的分类

    https://www.zhihu.com/question/32669957 目录 如何选购台式机CPU? 1. 英特尔处理器简介(本文) 1.1 聊聊Intel Tick-Tock 2. AMD处 ...

  2. UNP学习 ioctl操作

    一.ioctl函数 #include <unistd.h> int ioctl(int fd, int request, ... /* void * arg */); 返回:成功0,出错- ...

  3. hdu 6092 Rikka with Subset (集合计数,01背包)

    Problem Description As we know, Rikka is poor at math. Yuta is worrying about this situation, so he ...

  4. paper 138:qt安装及问题解决

    学习了很久的QT,遇到很多很多的问题,下面总结一下安装过程中遇到的问题吧, 1 下载QT 1)进入官网:https://www.qt.io/ 2)点击Download:https://www.qt.i ...

  5. linux系统一些信息(待整理)

    Linux查看物理CPU个数.核数.逻辑CPU个数 复制代码 # 总核数 = 物理CPU个数 X 每颗物理CPU的核数 # 总逻辑CPU数 = 物理CPU个数 X 每颗物理CPU的核数 X 超线程数 ...

  6. [六省联考2017]分手是祝愿 题解(期望dp)

    题目描述 B 君在玩一个游戏,这个游戏由 n 个灯和 n 个开关组成,给定这 n 个灯的初始状态,下标为从 1 到 n 的正整数. 每个灯有两个状态亮和灭,我们用 1 来表示这个灯是亮的,用 0 表示 ...

  7. Xcode 5 下的单元测试

    新版Xcode 5和Server发布以后,apple对单元测试的支持是越来越好了.从这一点看出apple对单元测试的也是越来越重视了. 这篇Blog就简单的介绍这集成化测试功能. Server更新后是 ...

  8. 国际C 语言乱码大赛(IOCCC )

    你也许听说过“国际C 语言乱码大赛(IOCCC )”,能获奖的人毫无疑问是世界顶级C 程序员.这是他们利用C 语言的特点极限挖掘的结果.下面这个例子就是网上广为流传的 一个经典作品:// 原始代码如下 ...

  9. Tomcat负载均衡、调优核心应用进阶学习笔记(一):tomcat文件目录、页面、架构组件详解、tomcat运行方式、组件介绍、tomcat管理

    文章目录 tomcat文件目录 bin conf lib logs temp webapps work 页面 架构组件详解 tomcat运行方式 组件介绍 tomcat管理 tomcat文件目录 ➜ ...

  10. Windows10系统下,如何彻底删除卸载MySQL

    本文介绍,在Windows10系统下,如何彻底删除卸载MySQL... 1>停止MySQL服务开始->所有应用->Windows管理工具->服务,将MySQL服务停止.2> ...