对于大神来说这题是水题。我搞这题花了快2天。

  伸展树的优点有什么,就是树不管你怎么旋转序列是不会改变得,并且你要使区间反转,只要把第k大的点转到根结点,那么它的左子树就是要交换的区间[l,r),然后交换左右

子树就可以了(中序),根结点的位置就是i+siz[ch[root][0]],i是处理完的结点个数,siz[ch[root][0]]就是左子树(需要旋转的个数)。 旋转可以用lazy思想标记,这样时间就为logn了。由于第k大的值已经处理完成,所以直接将根结点删除。

代码:

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<iostream>
  4. #include<algorithm>
  5. #define key_value ch[ch[root][1]][0]
  6. using namespace std;
  7. const int MAXN = ;
  8. struct node
  9. {
  10. int id;
  11. int v;
  12. }a[MAXN];
  13. int pre[MAXN],ch[MAXN][],siz[MAXN],lazy[MAXN],tot1,root;
  14. int s[MAXN],tot2,n;
  15. bool cmp(node fa,node fb)
  16. {
  17. if(fa.v != fb.v)
  18. return fa.v < fb.v;
  19. return fa.id < fb.id;
  20. }
  21. void Treavel(int x)
  22. {
  23. if(x)
  24. {
  25. Treavel(ch[x][]);
  26. printf("结点%2d:左儿子 %2d 右儿子 %2d 父结点 %2d size=%2d,lazy=%2d\n",x,ch[x][],ch[x][],pre[x],siz[x],lazy[x]);
  27. Treavel(ch[x][]);
  28. }
  29. }
  30. void debug()
  31. {
  32. printf("root:%d\n",root);
  33. Treavel(root);
  34. }
  35. void Newnode(int &rt,int pa,int k)
  36. {
  37. rt = k;
  38. pre[rt] = pa;
  39. siz[rt] = ;
  40. lazy[rt] = ;
  41. ch[rt][] = ch[rt][] = ;
  42. }
  43. void pushup(int rt)
  44. {
  45. siz[rt] = siz[ch[rt][]] + siz[ch[rt][]] + ;
  46. }
  47. void pushdown(int rt)
  48. {
  49. if(lazy[rt]){
  50. lazy[ch[rt][]] ^= ;
  51. lazy[ch[rt][]] ^= ;
  52. swap(ch[rt][],ch[rt][]);
  53. lazy[rt] = ;
  54. }
  55. }
  56. void build(int &rt,int l,int r,int pa)
  57. {
  58. if(l > r){
  59. return ;
  60. }
  61. int m = (l+r)/;
  62. Newnode(rt,pa,m);
  63. build(ch[rt][],l,m-,rt);
  64. build(ch[rt][],m+,r,rt);
  65. pushup(rt);
  66. }
  67. void Init()
  68. {
  69. root = tot1 = tot2 = ;
  70. ch[root][] = ch[root][] = siz[root] = lazy[root] = pre[root] = ;
  71. build(root,,n,);
  72. pushup(root);
  73. }
  74. void Rotate(int rt,int kind)
  75. {
  76. pushdown(pre[rt]);
  77. pushdown(rt);
  78. int y = pre[rt];
  79. ch[y][!kind] = ch[rt][kind];
  80. pre[ch[rt][kind]] = y;
  81. if(pre[y]){
  82. ch[pre[y]][ch[pre[y]][]==y] = rt;
  83. }
  84. pre[rt] = pre[y];
  85. ch[rt][kind] = y;
  86. pre[y] = rt;
  87. pushup(y);
  88. pushup(rt);
  89. }
  90. void splay(int rt,int goal)
  91. {
  92. pushdown(rt);
  93. while(pre[rt] != goal)
  94. {
  95. if(pre[pre[rt]] == goal){
  96. pushdown(pre[rt]);
  97. pushdown(rt);
  98. Rotate(rt,ch[pre[rt]][]==rt);
  99. }
  100. else {
  101. pushdown(pre[pre[rt]]);
  102. pushdown(pre[rt]);
  103. pushdown(rt);
  104. int y = pre[rt];
  105. int kind = ch[pre[y]][]==y;
  106. if(ch[y][kind] == rt){
  107. Rotate(rt,!kind);
  108. Rotate(rt,kind);
  109. }
  110. else {
  111. Rotate(y,kind);
  112. Rotate(rt,kind);
  113. }
  114. }
  115. }
  116. pushup(rt);
  117. if(goal == )
  118. root = rt;
  119.  
  120. }
  121. int Get_max(int rt)
  122. {
  123. pushdown(rt);
  124. while(ch[rt][]){
  125. rt = ch[rt][];
  126. pushdown(rt);
  127. }
  128. return rt;
  129. }
  130. void del(int root)
  131. {
  132. if(ch[root][] == ){
  133. root = ch[root][];
  134. pre[root] = ;
  135. }
  136. else {
  137. int t = Get_max(ch[root][]);
  138. splay(t,root);
  139. ch[t][] = ch[root][];
  140. pre[ch[root][]] = t;
  141. root = t;
  142. pre[root] = ;
  143. pushup(root);
  144. }
  145. }
  146. int main()
  147. {
  148. int i;
  149. while(~scanf("%d",&n),n)
  150. {
  151. for(i=; i<=n; i++){
  152. scanf("%d",&a[i].v);
  153. a[i].id = i;
  154. }
  155. sort(a+,a+n+,cmp);
  156. Init();
  157.  
  158. for(i=; i<=n; i++){
  159. //cout<<"test1: "<<endl;
  160.  
  161. splay(a[i].id,);
  162. // debug();
  163. lazy[ch[root][]] ^= ;
  164. printf("%d",i+siz[ch[root][]]);
  165. if(i < n)printf(" ");
  166. else printf("\n");
  167. del(root);
  168.  
  169. }
  170. }
  171. }

hdu1890 伸展树(区间反转)的更多相关文章

  1. 2018牛客网暑期ACM多校训练营(第三场) H - Shuffle Cards - [splay伸展树][区间移动][区间反转]

    题目链接:https://www.nowcoder.com/acm/contest/141/C 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K ...

  2. [置顶] hdu 1890 伸展树区间翻转

    题意: 给你n个数,每次先输出第i大的数的位置(如果有多个,选下标小的那个),然后每次将第i个位置到第i大的数所在位置之间的数进行翻转. 思路:输入的数组可能有多个相同的值,我们可以进行两次排序把数组 ...

  3. POJ 3468 A Simple Problem with Integers (伸展树区间更新求和操作 , 模板)

    伸展数最基本操作的模板,区间求和,区间更新.为了方便理解,特定附上一自己搞的搓图 这是样例中的数据输入后建成的树,其中的1,2是加入的边界顶点,数字代表节点编号,我们如果要对一段区间[l, r]进行操 ...

  4. POJ3580 SuperMemo splay伸展树,区间操作

    题意:实现一种数据结构,支持对一个数列的 6 种操作:第 x 个数到第 y 个数之间的数每个加 D:第 x 个数到第 y 个数之间全部数翻转:第 x 个数到第 y 个数之间的数,向后循环流动 c 次, ...

  5. PHP算法 《树形结构》 之 伸展树(1) - 基本概念

    伸展树的介绍 1.出处:http://dongxicheng.org/structure/splay-tree/ A. 概述 二叉查找树(Binary Search Tree,也叫二叉排序树,即Bin ...

  6. Splay伸展树入门(单点操作,区间维护)附例题模板

    Pps:终于学会了伸展树的区间操作,做一个完整的总结,总结一下自己的伸展树的单点操作和区间维护,顺便给未来的自己总结复习用. splay是一种平衡树,[平均]操作复杂度O(nlogn).首先平衡树先是 ...

  7. hdu 1754 splay tree伸展树 初战(单点更新,区间属性查询)

    题意:与区间查询点更新,点有20W个,询问区间的最大值.曾经用线段树,1000+ms,今天的伸展树,890没ms,差不多. 第一次学习伸展树,一共花了2个单位时间,感觉伸展树真很有用,也很好玩.现在只 ...

  8. 算法模板——splay区间反转 2

    实现功能:同splay区间反转 1(基于BZOJ3223 文艺平衡树) 这次改用了一个全新的模板(HansBug:琢磨了我大半天啊有木有),大大简化了程序,同时对于splay的功能也有所完善 这里面没 ...

  9. 伸展树Splay【非指针版】

    ·伸展树有以下基本操作(基于一道强大模板题:codevs维护队列): a[]读入的数组;id[]表示当前数组中的元素在树中节点的临时标号;fa[]当前节点的父节点的编号;c[][]类似于Trie,就是 ...

随机推荐

  1. 移动web页面使用的字体的思考

    前言 记得做PC端页面的时候,字体一般设置为微软雅黑,现在做起移动端页面来了,设计师们一般都还把字体设置为微软雅黑字体,但是做出来后,测试的时候发现页面中的字体不是微软雅黑,怎么办? 后来了解到的手机 ...

  2. CSS select样式优化 含jquery代码

    CSS 下拉选择菜单基本的CSS样式不怎么好看,通过一些简单的样式优化,可以得到如下图这样的: html结构如下: <div class="sel_wrap"> < ...

  3. leetcode - Merge Sorted Array (run time beats 100.00% of cpp submissions.)

    /** * Definition for singly-linked list. * struct ListNode { * int val; * ListNode *next; * ListNode ...

  4. Unity3D开发之搭建Mac OS开发环境

    运行图 首先上几张图 IOS模拟器 坚屏 横屏 打包任务 摸索了一上午,才搞定在模拟器中运行.至于在Iphone真机中运行,虽然有开发者证书,目前还没在Xcode中配置好. 我今天第一次接触并使用MA ...

  5. java 12-3 StringBuffer的添加和删除功能

    1. StringBuffer的添加功能: public StringBuffer append(String str):可以把任意类型数据添加到字符串缓冲区里面,并返回字符串缓冲区本身 public ...

  6. Swift中的Masonry第三方库——SnapKit

        在OC开发时我常用一个名叫Masonry的第三方Autolayout库,在转Swift后发现虽然Swift可以混编OC,但总感觉有些麻烦,在Github上发现了这个叫做SnapKit的第三方库 ...

  7. 30Springd的包扫描——<context:component-scan base-package=” ”/>

    在context中配置 如:在base-package指明一个包: <context:component-scan base-package="cn.edu.dao"/> ...

  8. CSS规则的执行顺序(转)

    你对CSS规则的执行顺序是否了解,这里和大家分享一下,若两条规则具有相同的权值.起源及特殊性,那在样式表中最后出现的规则优先. 1.CSS规则之特殊性 首先来看一下这个例子将会发生的情形: <s ...

  9. git config配置文件

    设置 git status的颜色. git config --global color.status auto 一.Git已经在你的系统中了,你会做一些事情来客户化你的Git环境.你只需要做这些设置一 ...

  10. mousedos网络批量部署xp

    小时候对这个东西很好奇,不知道什么原理.一直觉得很好玩.现在研究了下,总结如下 软件的操作步骤很讲究,稍微不慎,则就需要重新来过 知识点: 1,掌握诺顿ghost分区为gh文件 2,学会清理至一个干净 ...