题目描述

网址:https://www.luogu.org/problemnew/show/3721
大意:
有一颗单旋Splay(Spaly),以key值为优先度,总共有5个操作。

  • [1] 插入一个节点,需返还插入后此节点的深度。
  • [2] 把最小点单旋到根,需要返还旋转前此点深度。
  • [3] 把最大点单旋到根,需要返还旋转前此点深度。
  • [4] 把最小点单旋到根,然后删除根,需要返还旋转前此点深度。
  • [5] 把最大点单旋到根,然后删除根,需要返还旋转前此点深度。
    总共有M个操作,数据范围:M <= 100000

题目解法

题目提出的为splay算法,那么正解肯定不是splay。
观察到所有旋转操作都是对极值进行操作的,又是单旋,手玩一下就会发现树的形态基本不改变。
所以,我们可以建立一棵普通二叉树,由于每次修改的点都为常数个。
所以对于修改操作,手动修改即可。
然后考虑查询深度的问题,可以用LCT维护,修改普通二叉树对应的Link、Cut即可维持树的形态。
用LCT查询点u的深度,只需要把u到root的路径变为重路径,查询这棵辅助树的大小即可
最后是插入操作。用set可快速查询前驱与后继。
可以发现一定是插在两者中深度较大的下面,与前面一样直接修改即可。

实现代码

#include<bits/stdc++.h>
#define ll long long
#define RG register
#define IL inline
#define ls(x) ch[x][0]
#define rs(x) ch[x][1]
#define maxn 210000
using namespace std;

IL int gi(){  RG int date = 0, m = 1;  RG char ch = 0;
    while(ch!='-'&&(ch<'0'||ch>'9'))ch = getchar();
        if(ch == '-'){m = -1; ch = getchar();}
    while(ch>='0' && ch<='9')
           {date=date*10+ch-'0'; ch = getchar();}
        return date*m;
}

int root,cnt,M;
int Stk[maxn],ch[maxn][2],fa[maxn],rev[maxn],sz[maxn],fth[maxn];
int tr[maxn][2];

IL bool Son(int x){ return rs(fa[x]) == x; }
IL bool Isroot(int x){return (ls(fa[x])!=x && rs(fa[x])!=x) ; }
IL void PushUp(int x){ sz[x] = 1+sz[ls(x)]+sz[rs(x)]; }
IL void PushDown(int x){
    if(!rev[x])return; rev[x] = 0;
    rev[ls(x)]^=1; rev[rs(x)]^=1;  swap(ls(x) , rs(x));
}

IL void Rot(RG int x){
    RG int y = fa[x],z = fa[y],c = Son(x);
    if(!Isroot(y))ch[z][Son(y)] = x; fa[x] = z;
    ch[y][c] = ch[x][!c]; fa[ch[y][c]] = y;
    ch[x][!c] = y; fa[y] = x; PushUp(y);
}
IL void Splay(RG int x){
    RG int top = 0; Stk[++top] = x;
    for(RG int i = x; !Isroot(i); i = fa[i])Stk[++top] = fa[i];
    while(top)PushDown(Stk[top--]);
    for(RG int y = fa[x]; !Isroot(x); Rot(x),y = fa[x])
        if(!Isroot(y))Son(x) ^ Son(y) ? Rot(x) : Rot(y);
    PushUp(x);

}

IL void Access(int x){
    for(RG int y = 0; x; y = x,x = fa[x])
         Splay(x),ch[x][1] = y,PushUp(x);
}
IL void Makeroot(int x){Access(x);Splay(x); rev[x]^=1; }
IL void Split(int x,int y){ Makeroot(x); Access(y); Splay(y); }
IL void Link(int x,int y){ if(!x||!y)return; Makeroot(x); fa[x] = y; }
IL void Cut(int x,int y){
    if(!x||!y)return;
    Split(x,y); rs(y) = fa[x] = 0;
    PushUp(x); PushUp(y);
}
IL int Query(RG int x){Makeroot(root); Access(x); Splay(x); return sz[x];}  

set<int>S; set<int>::iterator bf,aft,it;
map<int,int>mp;
IL void NewNode(){cnt++; fa[cnt] = 0; sz[cnt] = 1;}

IL int Insert(){
    RG int data = gi(); S.insert(data);
    RG int dep,cs,dep1=0,dep2=0,v1,v2;
    bf = S.find(data); aft = bf; aft++;
    NewNode(); mp[data] = cnt;
    if(!root){root = cnt; return 1;}
    if(aft != S.end()){ v1 = mp[*aft]; dep1 = Query(v1); }
    if(bf !=S.begin()){bf--; v2 = mp[*bf]; dep2 = Query(v2);}
    cs = (dep1>dep2) ? v1:v2; dep = max(dep1,dep2);
    if(cs==v1)tr[cs][0] = cnt,fth[cnt] = cs;
    if(cs==v2)tr[cs][1] = cnt,fth[cnt] = cs;
    Link(cs,cnt);  return dep+1;
}

IL int Findmin1(){
    it = S.begin(); RG int p = mp[*it],dep;
    dep = Query(p);
    RG int x = p,sn = tr[x][1],ft = fth[x];
    if(x == root)return 1;
    Cut(x,sn); Cut(ft,x); Link(x,root); Link(ft,sn);
    fth[x] = 0; tr[x][1] = root; fth[root] = x; tr[ft][0] = sn; fth[sn] = ft;
    root = x; return dep;
}

IL int Findmax1(){
    it = S.end(); it--; RG int p = mp[*it],dep;
    dep = Query(p);
    RG int x = p,sn = tr[x][0],ft = fth[x];
    if(x == root)return 1;
    Cut(x,sn); Cut(ft,x); Link(x,root); Link(ft,sn);
    fth[x] = 0; tr[x][0] = root; fth[root] = x; tr[ft][1] = sn; fth[sn] = ft;
    root = x; return dep;
}

IL int Findmin2(){
    RG int dep = Findmin1(),rt = root,sn = tr[root][1];
    fth[sn] = 0;  Cut(root,sn);
    root = sn;
    tr[rt][1] = tr[rt][0] = 0;
    it = S.begin(); S.erase(it);
    return dep;
}

IL int Findmax2(){
    RG int dep = Findmax1(),rt = root,sn = tr[root][0];
    fth[sn] = 0;  Cut(root,sn);
    root = sn;
    tr[rt][1] = tr[rt][0] = 0;
    it = S.end(); it--; S.erase(it);
    return dep;
}

IL void Work(){
    RG int c = gi();
    if(c == 1)printf("%d\n",Insert());
    else if(c == 2)printf("%d\n",Findmin1());
    else if(c == 3)printf("%d\n",Findmax1());
    else if(c == 4)printf("%d\n",Findmin2());
    else if(c == 5)printf("%d\n",Findmax2());
}

int main()
{
    freopen("testdate.in","r",stdin);
    M = gi();
    root = 0; while(M--)Work();
    return 0;
}

HNOI2017 单旋的更多相关文章

  1. bzoj 4825: [Hnoi2017]单旋 [lct]

    4825: [Hnoi2017]单旋 题意:有趣的spaly hnoi2017刚出来我就去做,当时这题作死用了ett,调了5节课没做出来然后发现好像直接用lct就行了然后弃掉了... md用lct不知 ...

  2. 【LG3721】[HNOI2017]单旋

    [LG3721][HNOI2017]单旋 题面 洛谷 题解 20pts 直接模拟\(spaly\)的过程即可. 100pts 可以发现单旋最大.最小值到根,手玩是有显然规律的,发现只需要几次\(lin ...

  3. 4825: [Hnoi2017]单旋

    4825: [Hnoi2017]单旋 链接 分析: 以后采取更保险的方式写代码!!!81行本来以为不特判也可以,然后就总是比答案大1,甚至出现负数,调啊调啊调啊调~~~ 只会旋转最大值和最小值,以最小 ...

  4. [BZOJ4825][HNOI2017]单旋(线段树+Splay)

    4825: [Hnoi2017]单旋 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 667  Solved: 342[Submit][Status][ ...

  5. 【BZOJ4825】[Hnoi2017]单旋 线段树+set

    [BZOJ4825][Hnoi2017]单旋 Description H 国是一个热爱写代码的国家,那里的人们很小去学校学习写各种各样的数据结构.伸展树(splay)是一种数据结构,因为代码好写,功能 ...

  6. bzoj4825 [Hnoi2017]单旋

    Description H 国是一个热爱写代码的国家,那里的人们很小去学校学习写各种各样的数据结构.伸展树(splay)是一种数据结构,因为代码好写,功能多,效率高,掌握这种数据结构成为了 H 国的必 ...

  7. BZOJ:4825: [Hnoi2017]单旋

    Description H 国是一个热爱写代码的国家,那里的人们很小去学校学习写各种各样的数据结构.伸展树(splay)是一种数据结构,因为代码好写,功能多,效率高,掌握这种数据结构成为了 H 国的必 ...

  8. HNOI2017单旋

    单旋 这道题做法贼多,LCT,splay,线段树什么的貌似都行. 像我这种渣渣只会线段树了(高级数据结构学了也不会用). 首先离线所有操作,因为不会有两个点值重复,所以直接离散. 一颗线段树来维护所有 ...

  9. P3721 [AH2017/HNOI2017]单旋

    题目:https://www.luogu.org/problemnew/show/P3721 手玩一下即可AC此题. 结论:插入x后,x要么会成为x的前驱的右儿子,要么成为x的后继的左儿子,这取决于它 ...

随机推荐

  1. 炸金花的JS实现从0开始之 -------现在什么都不会(1)

    新年结束了.回想起来唯一留下乐趣的就是在家和朋友玩玩炸金花. 遂有此文. 对不起,我这时候还没有思路. 让我捋一捋. ... ... 捋一捋啊... ... 好了.今天先这样吧: (1)先整理出所有的 ...

  2. visual studio code右侧的预览面板能关闭吗?

    https://segmentfault.com/q/1010000010082399   "editor.minimap.enabled":false

  3. [转]【C#】分享一个弹出浮动层,像右键菜单那样召即来挥则去

    适用于:.net2.0+ Winform项目 背景: 有时候我们需要开一个简单的窗口来做一些事,例如输入一些东西.点选一个item之类的,可能像这样: 完了返回原窗体并获取刚刚的输入,这样做并没有什么 ...

  4. 深入java虚拟机学习 -- 类的加载机制

    当看到"类的加载机制",肯定很多人都在想我平时也不接触啊,工作中无非就是写代码,不会了可以百度,至于类,jvm是怎么加载的我一点也不需要关心.在我刚开始工作的时候也觉得这些底层的内 ...

  5. linux 安装 sftp

    1,sftp:登陆命令 Xshell:\> sftp root@192.168.159.128 Connecting to 192.168.159.128:22... Connection es ...

  6. Delphi 添加外部Form单元的方法!

    我用到的环境是 RAD Studio 10.2.2 有时候,需要把某个Form单元  添加到其他的工程!  此时,如果直接添加或者拖拉 .pas单元到目标工程,是无法把.pas包含的Form添加进去的 ...

  7. try{}里有一个 return 语句,那么紧跟在这个 try 后的 finally {}里的 code 会 不会被执行,什么时候被执行,在 return 前还是后?

    这是一道面试题,首先finally{}里面的code肯定是会执行的,至于在return前还是后, 看答案说的是在return后执行,我觉得不对,百度了一下,有说return前的,有说return后的, ...

  8. CodeForces-731B

    如果当天有m支队伍,昨天选择了k个B方案,那么今天还需要买m-k个披萨,如果m-k是奇数,那就先买一种B,剩下的全部买A,如果是偶数,全部买A.如果中途出现只有0支队伍,然而昨天却买了一次B,那么直接 ...

  9. 决策树--ID3 算法(一)

    Contents      1. 决策树的基本认识      2. ID3算法介绍      3. 信息熵与信息增益      4. ID3算法的C++实现 1. 决策树的基本认识    决策树是一种 ...

  10. java:产生小数位数为2的随机概率,使得和为1

    public static List<InstSec> setDataSec(List<String> instno) { List<InstSec> result ...