题目描述

请写一个程序,要求维护一个数列,支持以下 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)。

代码

  1. #include <cstdio>
  2. #include <cstdlib>
  3. #define rt ch[0][0]
  4. const int N=500000,S=N+100,inf=(1<<25)-1;
  5. 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];
  6. void rd(int &x)
  7. {
  8. x=0;char c=getchar(),t=1;
  9. while (c!='-' && (c<'0' || c>'9')) c=getchar();
  10. if (c=='-') t=-1,c=getchar();
  11. while (c>='0' && c<='9') x=(x<<3)+(x<<1)+(c-'0'),c=getchar();
  12. x*=t;
  13. }
  14. void init()
  15. {
  16. for (int i=0;i<=N;i++)
  17. bin[i]=i;
  18. bin[0]=1;
  19. rt=1;
  20. ls[0]=rs[0]=mx[0]=-inf;
  21. sz[0]=0;
  22. bin[0]=2;//use 1
  23. sz[1]=2;ch[1][1]=2;fa[1]=0;a[1]=-inf;add[1]=inf;
  24. bin[0]=3;//use 2
  25. sz[2]=1;fa[2]=1;add[2]=inf;a[2]=-inf;
  26. }
  27. inline int ma(int a,int b){if (a>b) return a;return b;}
  28. inline int ma3(int a,int b,int c){return ma(a,ma(b,c));}
  29. inline void swap(int &a,int &b){a^=b^=a^=b;}
  30. inline void up(int x)
  31. {
  32. int L=ch[x][0],R=ch[x][1];
  33. sz[x]=sz[L]+sz[R]+1;
  34. su[x]=su[L]+su[R]+a[x];
  35. ls[x]=ma3(ls[L],su[L]+a[x],su[L]+a[x]+ls[R]);
  36. rs[x]=ma3(rs[R],su[R]+a[x],su[R]+a[x]+rs[L]);
  37. int mid=ma(rs[L],0)+a[x]+ma(ls[R],0);
  38. mx[x]=ma3(mx[L],mx[R],mid);
  39. mx[x]=ma3(mx[x],ls[x],rs[x]);
  40. }
  41. inline void down(int x)
  42. {
  43. int &L=ch[x][0],&R=ch[x][1];
  44. if (rev[x])
  45. {
  46. if (L)
  47. {
  48. rev[L]^=1;
  49. swap(ch[L][0],ch[L][1]);
  50. swap(ls[L],rs[L]);
  51. }
  52. if (R)
  53. {
  54. rev[R]^=1;
  55. swap(ch[R][0],ch[R][1]);
  56. swap(ls[R],rs[R]);
  57. }
  58. rev[x]=0;
  59. }
  60. if (add[x]!=inf)
  61. {
  62. if (L)
  63. {
  64. add[L]=add[x];
  65. su[L]=add[L]*sz[L];
  66. a[L]=add[L];
  67. if (add[x]>0)
  68. ls[L]=rs[L]=mx[L]=su[L];
  69. else
  70. ls[L]=rs[L]=mx[L]=a[L];
  71. }
  72. if (R)
  73. {
  74. add[R]=add[x];
  75. su[R]=add[R]*sz[R];
  76. a[R]=add[R];
  77. if (add[x]>0)
  78. ls[R]=rs[R]=mx[R]=su[R];
  79. else
  80. ls[R]=rs[R]=mx[R]=a[R];
  81. }
  82. add[x]=inf;
  83. }
  84. }
  85. inline bool lor(int x){return ch[fa[x]][1]==x;}
  86. inline void link(int x,int fat,int o){fa[x]=fat;ch[fat][o]=x;}
  87. inline void rotate(int x)
  88. {
  89. int y=fa[x],r=fa[y];
  90. down(y);down(x);
  91. int rson=lor(y),yson=lor(x);
  92. link(ch[x][yson^1],y,yson);
  93. link(y,x,yson^1);
  94. link(x,r,rson);
  95. up(y);up(x);
  96. }
  97. void splay(int x,int to)
  98. {
  99. to=fa[to];
  100. while (fa[x]!=to)
  101. {
  102. if (fa[fa[x]]==to) rotate(x);
  103. else if (lor(x)==lor(fa[x])) rotate(fa[x]),rotate(x);
  104. else rotate(x),rotate(x);
  105. }
  106. }
  107. void build(int &k,int fat,int l,int r)
  108. {
  109. if (l>r)
  110. {
  111. k=0;
  112. return;
  113. }
  114. k=bin[bin[0]++];
  115. fa[k]=fat;
  116. int mid=(l+r)>>1;
  117. build(ch[k][0],k,l,mid-1);
  118. rd(a[k]);
  119. add[k]=inf;rev[k]=0;
  120. build(ch[k][1],k,mid+1,r);
  121. up(k);
  122. }
  123. void recycle(int k)
  124. {
  125. if (!k) return;
  126. recycle(ch[k][0]);
  127. a[k]=su[k]=ls[k]=rs[k]=mx[k]=fa[k]=add[k]=rev[k]=0;
  128. bin[--bin[0]]=k;
  129. recycle(ch[k][1]);
  130. ch[k][0]=ch[k][1]=0;
  131. }
  132. int find(int x)
  133. {
  134. int o=rt;
  135. while (o)
  136. {
  137. down(o);
  138. if (sz[ch[o][0]]+1==x)
  139. break;
  140. if (x<=sz[ch[o][0]])
  141. o=ch[o][0];
  142. else x-=sz[ch[o][0]]+1,o=ch[o][1];
  143. }
  144. return o;
  145. }
  146. void insert(int pos,int tot)
  147. {
  148. pos++;
  149. int bg=find(pos),ed=find(pos+1);
  150. splay(bg,rt);
  151. splay(ed,ch[bg][1]);
  152. build(ch[ed][0],ed,1,tot);
  153. up(ed);up(bg);
  154. }
  155. void del(int pos,int tot)
  156. {
  157. int bg=find(pos),ed=find(pos+tot+1);
  158. splay(bg,rt);splay(ed,ch[bg][1]);
  159. recycle(ch[ed][0]);
  160. ch[ed][0]=0;
  161. up(ed);up(bg);
  162. }
  163. void make_same(int pos,int tot,int x)
  164. {
  165. int bg=find(pos),ed=find(pos+tot+1);
  166. splay(bg,rt);splay(ed,ch[bg][1]);
  167. int o=ch[ed][0];
  168. add[o]=x;su[o]=x*sz[o];a[o]=x;
  169. if (x>0) ls[o]=rs[o]=mx[o]=su[o];
  170. else ls[o]=rs[o]=mx[o]=a[o];
  171. up(ed);up(bg);
  172. }
  173. void reverse(int pos,int tot)
  174. {
  175. int bg=find(pos),ed=find(pos+tot+1);
  176. splay(bg,rt);splay(ed,ch[bg][1]);
  177. down(bg);down(ed);
  178. int o=ch[ed][0];
  179. rev[o]=1;
  180. swap(ch[o][0],ch[o][1]);
  181. swap(ls[o],rs[o]);
  182. up(ed);up(bg);
  183. }
  184. void getsum(int pos,int tot)
  185. {
  186. int bg=find(pos),ed=find(pos+tot+1);
  187. splay(bg,rt);
  188. splay(ed,ch[bg][1]);
  189. int o=ch[ed][0];
  190. printf("%d\n",su[o]);
  191. }
  192. int main()
  193. {
  194. rd(n);rd(m);
  195. init();
  196. build(ch[2][0],2,1,n);up(2);up(1);
  197. char op[10];int tot,pos,x;
  198. while (m--)
  199. {
  200. scanf("%s",op);
  201. if (op[2]=='X')//max_sum
  202. printf("%d\n",mx[rt]);
  203. else
  204. {
  205. rd(pos);rd(tot);
  206. if (op[2]=='S')//Insert
  207. insert(pos,tot);
  208. else if (op[2]=='L')
  209. del(pos,tot);
  210. else if ('K'==op[2])
  211. {
  212. scanf("%d",&x);
  213. make_same(pos,tot,x);
  214. }
  215. else if ('V'==op[2])
  216. reverse(pos,tot);
  217. else getsum(pos,tot);
  218. }
  219. }
  220. return 0;
  221. }

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. MariaDB 创建表

    在本章中,我们将学习如何创建表. 在创建表之前,首先确定其名称,字段名称和字段定义. 以下是表创建的一般语法: CREATE TABLE table_name (column_name column_ ...

  2. jquery实现回车键登录/搜索等确认功能

    button按钮提交方式: $('#search').click(function() { get_table(); }); //keyCode=13是回车键,设置回车键提交 $("body ...

  3. XAMPP安装和配置

    一.XAMPP安装: 下载地址:https://www.apachefriends.org/zh_cn/index.html 二.修改MySQL数据库 1.更改Apache中数据库端口号 保存后重新启 ...

  4. sync.Once.Do(f func())

    sync.Once.Do(f func())是一个挺有趣的东西,能保证once只执行一次,无论你是否更换once.Do(xx)这里的方法,这个sync.Once块只会执行一次. package mai ...

  5. LiveTelecast直播平台技术图谱skill-map

    #直播平台技术图谱 ----##直播 ----###采集- **iOS** * HTTP Live Streaming * DirectShow- **Android** * setPreviewCa ...

  6. JQ-jQuery-Ajax:jQuery Ajax 操作函数

    ylbtech-JQ-jQuery-Ajax:jQuery Ajax 操作函数 1.返回顶部 1. jQuery Ajax 操作函数 jQuery 库拥有完整的 Ajax 兼容套件.其中的函数和方法允 ...

  7. 02 java语言基础

    常量:字面值常量(字符串,字符,整数,小数,布尔,null),自定义常量,''这个不是字符常量,""这个是字符串常量 进制: 02.01_Java语言基础(常量的概述和使用) A: ...

  8. 201⑨湘潭邀请赛 Chika and Friendly Pairs(HDU6534)

    原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=6534 题意: 给你一个数组,对于第i个数来说,如果存在一个位置j,使得j>i并且a[j]-k&l ...

  9. InnoDB B树 锁

    InnoDB B树 叶子=>主键+数记录非叶子=>主键1+主键3...主键4 事务和行锁 索引项加锁 相等条件来访问更新数据,避免使用范围条件 (1)InnoDB的行销是基于索引实现的,如 ...

  10. java.sql.BatchUpdateException: ORA-01861: 文字与格式字符串不匹配

    解决: to_date(#runtime#,'yyyy-MM-dd HH24:mi:ss'), <!-- 执行时间:DATE -->