题目

\(\rm splay\)水平太差,于是得手玩一下才能发现规律

首先插入一个数,其肯定会成为其前驱的右儿子或者是后继的左儿子,进一步手玩发现前驱的右儿子或者是后继的左儿子一定只有一个是空的,我们找到这个空位置插入就好了

于是我们需要一个\(\rm std::set\)来查找前驱后继,同时我们还需要维护每个点的左右儿子和父亲

继续手玩发现由于只有对最大值和最小值的操作,所以对\(\rm splay\)的结构影响很小,于是这个过程中也能维护每个节点的父亲的左右儿子

深度看起来不能用几个数组来维护了,于是我们直接用lct维护这棵树的形态,深度查一下到当前根的路径上的节点个数即可

代码

#include<bits/stdc++.h>
#define re register
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
const int maxn=1e5+5;
struct LinkCutTree {
int fa[maxn],ch[maxn][2],sz[maxn],rev[maxn];
int st[maxn],top;
inline int nrt(int x) {return ch[fa[x]][1]==x||ch[fa[x]][0]==x;}
inline void pushup(int x) {sz[x]=sz[ch[x][0]]+1+sz[ch[x][1]];}
inline void rotate(int x) {
int y=fa[x],z=fa[y],w=ch[y][1]==x,k=ch[x][w^1];
if(nrt(y)) ch[z][ch[z][1]==y]=x;
ch[x][w^1]=y,ch[y][w]=k;
pushup(y),pushup(x);fa[k]=y,fa[y]=x,fa[x]=z;
}
inline void work(int x) {rev[x]^=1;std::swap(ch[x][0],ch[x][1]);}
inline void pushdown(int x) {
if(!rev[x]) return;rev[x]=0;
if(ch[x][0]) work(ch[x][0]);
if(ch[x][1]) work(ch[x][1]);
}
inline void splay(int x) {
int y=x;top=0;
st[++top]=x;
while(nrt(y)) y=fa[y],st[++top]=y;
while(top) pushdown(st[top--]);
while(nrt(x)) {
int y=fa[x];
if(nrt(y)) rotate((ch[y][1]==x)^(ch[fa[y]][1]==y)?x:y);
rotate(x);
}
}
inline void access(int x) {
for(re int y=0;x;y=x,x=fa[x])
splay(x),ch[x][1]=y,pushup(x);
}
inline void mrt(int x) {
access(x);splay(x);work(x);
}
inline void link(int x,int y) {
mrt(x);fa[x]=y;
}
inline void split(int x,int y) {
mrt(x),access(y),splay(y);
}
inline void cut(int x,int y) {
split(x,y);fa[x]=ch[y][0]=0;pushup(y);
}
inline int dis(int x,int y) {
split(x,y);return sz[y];
}
}lct;
struct RBT {
std::set<int> s;
#define It std::set<int>::iterator
inline int pre(int x) {
It it=s.find(x);
if(it==s.begin()) return 0;
--it;return *it;
}
inline int nxt(int x) {
It it=s.find(x);++it;
if(it==s.end()) return 0;
return *it;
}
inline void del(int x) {s.erase(x);}
inline void ins(int x) {s.insert(x);}
inline int Gmin() {It it=s.begin();return *it;}
inline int Gmax() {It it=s.end();--it;return *it;}
inline int empty() {return s.empty();}
}s;
int n,m,rt;
int a[maxn],b[maxn],son[maxn][2],op[maxn],fa[maxn];
inline int find(int x) {
int l=1,r=n;
while(l<=r) {
int mid=l+r>>1;
if(b[mid]==x) return mid;
if(b[mid]<x) l=mid+1;else r=mid-1;
}
return 0;
}
inline int ins(int x) {
s.ins(x);
if(!rt) {fa[x]=0;rt=x;return 1;}
int Pre=s.pre(x);
if(Pre&&!son[Pre][1])
lct.link(Pre,x),son[Pre][1]=x,fa[x]=Pre;
else {
int Nxt=s.nxt(x);
if(Nxt&&!son[Nxt][0])
lct.link(Nxt,x),son[Nxt][0]=x,fa[x]=Nxt;
}
return lct.dis(rt,x);
}
inline void move_min() {
int x=s.Gmin();
printf("%d\n",lct.dis(x,rt));
if(x==rt) return;
lct.cut(fa[x],x);
if(son[x][1]) lct.cut(x,son[x][1]);
lct.link(x,rt);
if(son[x][1]) lct.link(fa[x],son[x][1]);
fa[son[x][1]]=fa[x];fa[rt]=x;
son[fa[x]][0]=son[x][1];
son[x][1]=rt;fa[x]=0;rt=x;
}
inline void move_max() {
int x=s.Gmax();
printf("%d\n",lct.dis(x,rt));
if(x==rt) return;
lct.cut(fa[x],x);
if(son[x][0]) lct.cut(x,son[x][0]);
lct.link(rt,x);
if(son[x][0]) lct.link(fa[x],son[x][0]);
fa[son[x][0]]=fa[x],fa[rt]=x;
son[fa[x]][1]=son[x][0];
son[x][0]=rt,fa[x]=0;rt=x;
}
inline void pop(int x) {
if(son[x][0]) lct.cut(x,son[x][0]),rt=son[x][0];
if(son[x][1]) lct.cut(x,son[x][1]),rt=son[x][1];
son[x][0]=son[x][1]=fa[x]=0;
s.del(x);if(s.empty()) rt=0;
}
int main() {
m=read();
for(re int i=1;i<=m;i++) {
op[i]=read();
if(op[i]==1) a[i]=read(),b[++n]=a[i];
}
std::sort(b+1,b+n+1);
for(re int i=1;i<=m;i++) {
if(op[i]==1) printf("%d\n",ins(find(a[i])));
if(op[i]==2) move_min();
if(op[i]==3) move_max();
if(op[i]==4) move_min(),pop(rt);
if(op[i]==5) move_max(),pop(rt);
}
return 0;
}

[AH2017/HNOI2017]单旋的更多相关文章

  1. P3721 [AH2017/HNOI2017]单旋

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

  2. 洛谷 P3721 - [AH2017/HNOI2017]单旋(LCT)

    洛谷题面传送门 终于调出来这道题了,写篇题解( 首先碰到这样的题我们肯定要考虑每种操作会对树的形态产生怎样的影响: 插入操作:对于 BST 有一个性质是,当你插入一个节点时,其在 BST 上的父亲肯定 ...

  3. luogu P3721 [AH2017/HNOI2017]单旋

    传送门 \(Spaly:\)??? 考虑在暴力模拟的基础上优化 如果要插入一个数,那么根据二叉查找树的性质,这个点一定插在他的前驱的右子树或者是后继的左子树,可以利用set维护当前树里面的数,方便查找 ...

  4. 洛谷P3721 [AH2017/HNOI2017]单旋(线段树 set spaly)

    题意 题目链接 Sol 这题好毒瘤啊.. 首先要观察到几个性质: 将最小值旋转到根相当于把右子树变为祖先的左子树,然后将原来的根变为当前最小值 上述操作对深度的影响相当于右子树不变,其他的位置-1 然 ...

  5. [AH2017/HNOI2017] 单旋 - Splay

    Splay 暴力维护节点信息即可 #include<iostream> #include<cstdio> #include<cstring> #include< ...

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

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

  7. 【LG3721】[HNOI2017]单旋

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

  8. 4825: [Hnoi2017]单旋

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

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

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

随机推荐

  1. go语言将函数作为参数传递

    Go语言函数作为参数传递,目前给我的感觉几乎和C/C++一致.非常的灵活. import "fmt" import "time" func goFunc1(f  ...

  2. arttemplate 后台返回的数据格式问题

    1. 2.JSON.parse() 方法用于将一个 JSON 字符串转换为对象. $("body").on("click","#analyze&quo ...

  3. Git 学习第二天(一)

    继续昨天的学习. 回顾一下,昨天我们安装了git 并创建了登录名及邮箱,还向git仓库提交了一个readme.txt的文本文件 下面,我们来修改下这个文件,将内容改为 Git is a distrib ...

  4. 前端(十五)—— JavaScript事件:绑定事件方式、事件的冒泡和默认事件、鼠标事件、键盘事件、表单 事件、文档事件、图片事件、页面事件

    JS事件:绑定事件方式.事件的冒泡和默认事件.鼠标事件.键盘事件.表单 事件.文档事件.图片事件.页面事件 一.事件的两种绑定方式 1.on事件绑定方式 document.onclick = func ...

  5. java.lang.Double.byteValue() 方法

    java.lang.Double.byteValue() 方法(通过转换成一个字节)返回此Double为一个字节的值. 声明 以下是java.lang.Double.byteValue()方法的声明 ...

  6. js 中typeof 检测数据类型的时候需要注意的小细节

    博客搬迁给你带来的不便,敬请谅解! http://www.suanliutudousi.com/2017/10/26/typeof-%E6%A3%80%E6%B5%8B%E6%95%B0%E6%8D% ...

  7. JSON数组对象和JSON字符串的转化,map和JSON对象之间的转化

    这种用法包括前端和后端: 前端: 1. 转化为JSON对象方便操作 var jsonObj = JSON.parse(str); 得到的是一个json数组对象,可以通过 for (var p in j ...

  8. js面向对象(三)---组件开发

    一.对象的多种表现形式 1.提高对象的复用性 2.如何配置参数和默认参数 不知道该怎么描述,就直接上代码吧,下面做了2个例子,重点看整个组件的大体结构 用组件的方式做拖拽窗口,你可以狠狠的点击这里进行 ...

  9. 3-vim-打开和新建文件-02-删除交换文件

    vim的异常处理 如果vim异常退出,在磁盘上可能会保存有交换文件. 若使用vi编辑该文件时看到如下图信息,按下字母d就可以删除交换文件. 注意:输入命令操作的时候关闭输入法.

  10. Warshall算法和Floyd算法

    不用说这两位都是冷门算法……毕竟O(n^3)的时间复杂度算法在算法竞赛里基本算是被淘汰了……而且也没有在这个算法上继续衍生出其他的算法… 有兴趣的话:click here.. 话说学离散的时候曾经有个 ...