hdu1890 伸展树(区间反转)
对于大神来说这题是水题。我搞这题花了快2天。
伸展树的优点有什么,就是树不管你怎么旋转序列是不会改变得,并且你要使区间反转,只要把第k大的点转到根结点,那么它的左子树就是要交换的区间[l,r),然后交换左右
子树就可以了(中序),根结点的位置就是i+siz[ch[root][0]],i是处理完的结点个数,siz[ch[root][0]]就是左子树(需要旋转的个数)。 旋转可以用lazy思想标记,这样时间就为logn了。由于第k大的值已经处理完成,所以直接将根结点删除。
代码:
- #include<cstdio>
- #include<cstring>
- #include<iostream>
- #include<algorithm>
- #define key_value ch[ch[root][1]][0]
- using namespace std;
- const int MAXN = ;
- struct node
- {
- int id;
- int v;
- }a[MAXN];
- int pre[MAXN],ch[MAXN][],siz[MAXN],lazy[MAXN],tot1,root;
- int s[MAXN],tot2,n;
- bool cmp(node fa,node fb)
- {
- if(fa.v != fb.v)
- return fa.v < fb.v;
- return fa.id < fb.id;
- }
- void Treavel(int x)
- {
- if(x)
- {
- Treavel(ch[x][]);
- printf("结点%2d:左儿子 %2d 右儿子 %2d 父结点 %2d size=%2d,lazy=%2d\n",x,ch[x][],ch[x][],pre[x],siz[x],lazy[x]);
- Treavel(ch[x][]);
- }
- }
- void debug()
- {
- printf("root:%d\n",root);
- Treavel(root);
- }
- void Newnode(int &rt,int pa,int k)
- {
- rt = k;
- pre[rt] = pa;
- siz[rt] = ;
- lazy[rt] = ;
- ch[rt][] = ch[rt][] = ;
- }
- void pushup(int rt)
- {
- siz[rt] = siz[ch[rt][]] + siz[ch[rt][]] + ;
- }
- void pushdown(int rt)
- {
- if(lazy[rt]){
- lazy[ch[rt][]] ^= ;
- lazy[ch[rt][]] ^= ;
- swap(ch[rt][],ch[rt][]);
- lazy[rt] = ;
- }
- }
- void build(int &rt,int l,int r,int pa)
- {
- if(l > r){
- return ;
- }
- int m = (l+r)/;
- Newnode(rt,pa,m);
- build(ch[rt][],l,m-,rt);
- build(ch[rt][],m+,r,rt);
- pushup(rt);
- }
- void Init()
- {
- root = tot1 = tot2 = ;
- ch[root][] = ch[root][] = siz[root] = lazy[root] = pre[root] = ;
- build(root,,n,);
- pushup(root);
- }
- void Rotate(int rt,int kind)
- {
- pushdown(pre[rt]);
- pushdown(rt);
- int y = pre[rt];
- ch[y][!kind] = ch[rt][kind];
- pre[ch[rt][kind]] = y;
- if(pre[y]){
- ch[pre[y]][ch[pre[y]][]==y] = rt;
- }
- pre[rt] = pre[y];
- ch[rt][kind] = y;
- pre[y] = rt;
- pushup(y);
- pushup(rt);
- }
- void splay(int rt,int goal)
- {
- pushdown(rt);
- while(pre[rt] != goal)
- {
- if(pre[pre[rt]] == goal){
- pushdown(pre[rt]);
- pushdown(rt);
- Rotate(rt,ch[pre[rt]][]==rt);
- }
- else {
- pushdown(pre[pre[rt]]);
- pushdown(pre[rt]);
- pushdown(rt);
- int y = pre[rt];
- int kind = ch[pre[y]][]==y;
- if(ch[y][kind] == rt){
- Rotate(rt,!kind);
- Rotate(rt,kind);
- }
- else {
- Rotate(y,kind);
- Rotate(rt,kind);
- }
- }
- }
- pushup(rt);
- if(goal == )
- root = rt;
- }
- int Get_max(int rt)
- {
- pushdown(rt);
- while(ch[rt][]){
- rt = ch[rt][];
- pushdown(rt);
- }
- return rt;
- }
- void del(int root)
- {
- if(ch[root][] == ){
- root = ch[root][];
- pre[root] = ;
- }
- else {
- int t = Get_max(ch[root][]);
- splay(t,root);
- ch[t][] = ch[root][];
- pre[ch[root][]] = t;
- root = t;
- pre[root] = ;
- pushup(root);
- }
- }
- int main()
- {
- int i;
- while(~scanf("%d",&n),n)
- {
- for(i=; i<=n; i++){
- scanf("%d",&a[i].v);
- a[i].id = i;
- }
- sort(a+,a+n+,cmp);
- Init();
- for(i=; i<=n; i++){
- //cout<<"test1: "<<endl;
- splay(a[i].id,);
- // debug();
- lazy[ch[root][]] ^= ;
- printf("%d",i+siz[ch[root][]]);
- if(i < n)printf(" ");
- else printf("\n");
- del(root);
- }
- }
- }
hdu1890 伸展树(区间反转)的更多相关文章
- 2018牛客网暑期ACM多校训练营(第三场) H - Shuffle Cards - [splay伸展树][区间移动][区间反转]
题目链接:https://www.nowcoder.com/acm/contest/141/C 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K ...
- [置顶] hdu 1890 伸展树区间翻转
题意: 给你n个数,每次先输出第i大的数的位置(如果有多个,选下标小的那个),然后每次将第i个位置到第i大的数所在位置之间的数进行翻转. 思路:输入的数组可能有多个相同的值,我们可以进行两次排序把数组 ...
- POJ 3468 A Simple Problem with Integers (伸展树区间更新求和操作 , 模板)
伸展数最基本操作的模板,区间求和,区间更新.为了方便理解,特定附上一自己搞的搓图 这是样例中的数据输入后建成的树,其中的1,2是加入的边界顶点,数字代表节点编号,我们如果要对一段区间[l, r]进行操 ...
- POJ3580 SuperMemo splay伸展树,区间操作
题意:实现一种数据结构,支持对一个数列的 6 种操作:第 x 个数到第 y 个数之间的数每个加 D:第 x 个数到第 y 个数之间全部数翻转:第 x 个数到第 y 个数之间的数,向后循环流动 c 次, ...
- PHP算法 《树形结构》 之 伸展树(1) - 基本概念
伸展树的介绍 1.出处:http://dongxicheng.org/structure/splay-tree/ A. 概述 二叉查找树(Binary Search Tree,也叫二叉排序树,即Bin ...
- Splay伸展树入门(单点操作,区间维护)附例题模板
Pps:终于学会了伸展树的区间操作,做一个完整的总结,总结一下自己的伸展树的单点操作和区间维护,顺便给未来的自己总结复习用. splay是一种平衡树,[平均]操作复杂度O(nlogn).首先平衡树先是 ...
- hdu 1754 splay tree伸展树 初战(单点更新,区间属性查询)
题意:与区间查询点更新,点有20W个,询问区间的最大值.曾经用线段树,1000+ms,今天的伸展树,890没ms,差不多. 第一次学习伸展树,一共花了2个单位时间,感觉伸展树真很有用,也很好玩.现在只 ...
- 算法模板——splay区间反转 2
实现功能:同splay区间反转 1(基于BZOJ3223 文艺平衡树) 这次改用了一个全新的模板(HansBug:琢磨了我大半天啊有木有),大大简化了程序,同时对于splay的功能也有所完善 这里面没 ...
- 伸展树Splay【非指针版】
·伸展树有以下基本操作(基于一道强大模板题:codevs维护队列): a[]读入的数组;id[]表示当前数组中的元素在树中节点的临时标号;fa[]当前节点的父节点的编号;c[][]类似于Trie,就是 ...
随机推荐
- 移动web页面使用的字体的思考
前言 记得做PC端页面的时候,字体一般设置为微软雅黑,现在做起移动端页面来了,设计师们一般都还把字体设置为微软雅黑字体,但是做出来后,测试的时候发现页面中的字体不是微软雅黑,怎么办? 后来了解到的手机 ...
- CSS select样式优化 含jquery代码
CSS 下拉选择菜单基本的CSS样式不怎么好看,通过一些简单的样式优化,可以得到如下图这样的: html结构如下: <div class="sel_wrap"> < ...
- leetcode - Merge Sorted Array (run time beats 100.00% of cpp submissions.)
/** * Definition for singly-linked list. * struct ListNode { * int val; * ListNode *next; * ListNode ...
- Unity3D开发之搭建Mac OS开发环境
运行图 首先上几张图 IOS模拟器 坚屏 横屏 打包任务 摸索了一上午,才搞定在模拟器中运行.至于在Iphone真机中运行,虽然有开发者证书,目前还没在Xcode中配置好. 我今天第一次接触并使用MA ...
- java 12-3 StringBuffer的添加和删除功能
1. StringBuffer的添加功能: public StringBuffer append(String str):可以把任意类型数据添加到字符串缓冲区里面,并返回字符串缓冲区本身 public ...
- Swift中的Masonry第三方库——SnapKit
在OC开发时我常用一个名叫Masonry的第三方Autolayout库,在转Swift后发现虽然Swift可以混编OC,但总感觉有些麻烦,在Github上发现了这个叫做SnapKit的第三方库 ...
- 30Springd的包扫描——<context:component-scan base-package=” ”/>
在context中配置 如:在base-package指明一个包: <context:component-scan base-package="cn.edu.dao"/> ...
- CSS规则的执行顺序(转)
你对CSS规则的执行顺序是否了解,这里和大家分享一下,若两条规则具有相同的权值.起源及特殊性,那在样式表中最后出现的规则优先. 1.CSS规则之特殊性 首先来看一下这个例子将会发生的情形: <s ...
- git config配置文件
设置 git status的颜色. git config --global color.status auto 一.Git已经在你的系统中了,你会做一些事情来客户化你的Git环境.你只需要做这些设置一 ...
- mousedos网络批量部署xp
小时候对这个东西很好奇,不知道什么原理.一直觉得很好玩.现在研究了下,总结如下 软件的操作步骤很讲究,稍微不慎,则就需要重新来过 知识点: 1,掌握诺顿ghost分区为gh文件 2,学会清理至一个干净 ...