[AH2017/HNOI2017]单旋
\(\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]单旋的更多相关文章
- P3721 [AH2017/HNOI2017]单旋
题目:https://www.luogu.org/problemnew/show/P3721 手玩一下即可AC此题. 结论:插入x后,x要么会成为x的前驱的右儿子,要么成为x的后继的左儿子,这取决于它 ...
- 洛谷 P3721 - [AH2017/HNOI2017]单旋(LCT)
洛谷题面传送门 终于调出来这道题了,写篇题解( 首先碰到这样的题我们肯定要考虑每种操作会对树的形态产生怎样的影响: 插入操作:对于 BST 有一个性质是,当你插入一个节点时,其在 BST 上的父亲肯定 ...
- luogu P3721 [AH2017/HNOI2017]单旋
传送门 \(Spaly:\)??? 考虑在暴力模拟的基础上优化 如果要插入一个数,那么根据二叉查找树的性质,这个点一定插在他的前驱的右子树或者是后继的左子树,可以利用set维护当前树里面的数,方便查找 ...
- 洛谷P3721 [AH2017/HNOI2017]单旋(线段树 set spaly)
题意 题目链接 Sol 这题好毒瘤啊.. 首先要观察到几个性质: 将最小值旋转到根相当于把右子树变为祖先的左子树,然后将原来的根变为当前最小值 上述操作对深度的影响相当于右子树不变,其他的位置-1 然 ...
- [AH2017/HNOI2017] 单旋 - Splay
Splay 暴力维护节点信息即可 #include<iostream> #include<cstdio> #include<cstring> #include< ...
- bzoj 4825: [Hnoi2017]单旋 [lct]
4825: [Hnoi2017]单旋 题意:有趣的spaly hnoi2017刚出来我就去做,当时这题作死用了ett,调了5节课没做出来然后发现好像直接用lct就行了然后弃掉了... md用lct不知 ...
- 【LG3721】[HNOI2017]单旋
[LG3721][HNOI2017]单旋 题面 洛谷 题解 20pts 直接模拟\(spaly\)的过程即可. 100pts 可以发现单旋最大.最小值到根,手玩是有显然规律的,发现只需要几次\(lin ...
- 4825: [Hnoi2017]单旋
4825: [Hnoi2017]单旋 链接 分析: 以后采取更保险的方式写代码!!!81行本来以为不特判也可以,然后就总是比答案大1,甚至出现负数,调啊调啊调啊调~~~ 只会旋转最大值和最小值,以最小 ...
- [BZOJ4825][HNOI2017]单旋(线段树+Splay)
4825: [Hnoi2017]单旋 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 667 Solved: 342[Submit][Status][ ...
随机推荐
- (转)微信,QQ这类IM app怎么做——谈谈Websocket
转:http://www.cocoachina.com/ios/20160527/16482.html 前言 关于我和WebSocket的缘:我从大二在计算机网络课上听老师讲过之后,第一次使用就到了毕 ...
- 杂项-WebService:WebService
ylbtech-杂项-WebService:WebService Web service是一个平台独立的,低耦合的,自包含的.基于可编程的web的应用程序,可使用开放的XML(标准通用标记语言下的一个 ...
- continuation line under-indented for visual indent
continuation line under-indented for visual indent 问题:使用flake8检验代码规范时报错:continuation line under-inde ...
- Openstack组件实现原理 — Nova 体系结构
目录 目录 前文列表 Nova体系结构 虚拟机实例化流程 前文列表 Openstack组件部署 - Overview和前期环境准备 Openstack组建部署 - Environment of Con ...
- mkdir无法创建目录权限不够
https://idc.wanyunshuju.com/cym/40.html 由于在公司服务器上权限问题比较复杂,我们解决这个问题是寻求服务器端人员的意见.让他们授予权限/.
- 【HDOJ】P2058 The sum problem
题意很简单就是给你一个N和M,让你求在1-N的那些个子序列的值等于M 首先暴力法不解释,简单超时 再仔细想一想可以想到因为1-N是一个等差数列,可以运用我们曾经学过的只是来解决 假设开始的位置为s,结 ...
- python--reflect
一.反射 python 中用字符串的方式操作对象的相关属性,python 中一切皆对象,都可以使用反射 用eval 有安全隐患,用 反射就很安全 1.反射对象中的属性和方法 class A: a_cl ...
- 微信小程序开发显示城市天气
本案例实现动态显示城市天气的功能,案例效果如下: 首先分析制作的思路: 1.在app.json文件的pages数组里加上main文件夹和template(模板)文件夹的路径. 2.在main.js文件 ...
- keep-alive用法及(activated,deactivated生命周期)
<template> <div id="app"> <!-- <img src="./assets/logo.png"> ...
- 使用wireshark在windows平台下捕获HTTP协议数据包中的帐号密码信息
1.打开wireshark软件,从Interface List中选择相应的网卡,例如我的PC机上是“本地连接”,然后选择”Start”启动抓包程序. 2.打开学校主页,输入账号和密码登录校内邮箱. 3 ...