洛谷.3369.[模板]普通平衡树(Splay)
题目链接
第一次写(2017.11.7):
#include<cstdio>
#include<cctype>
using namespace std;
const int N=1e5+5,INF=1e9;
int size,root,t[N],sz[N],son[N][2],fa[N],cnt[N];
inline int read()
{
int now=0,f=1;register char c=getchar();
for(;!isdigit(c);c=getchar())
if(c=='-') f=-1;
for(;isdigit(c);now=now*10+c-'0',c=getchar());
return now*f;
}
inline void Update(int rt)
{
sz[rt]= sz[son[rt][0]]+sz[son[rt][1]]+cnt[rt];
}
void Rotate(int x,int &k)
{
int a=fa[x],b=fa[a],l=(son[a][1]==x),r=l^1;
if(a==k) k=x;
else son[b][son[b][1]==a]=x;
fa[a]=x, fa[x]=b, fa[son[x][r]]=a,
son[a][l]=son[x][r], son[x][r]=a;
Update(a),Update(x);
}
void Splay(int x,int &rt)
{
int a,b;
while(x!=rt)
{
a=fa[x], b=fa[a];
if(a!=rt)
{
if((son[a][0]==x)^(son[b][0]==a)) Rotate(x,rt);
else Rotate(a,rt);
}
Rotate(x,rt);
}
}
void Insert(int k,int v)
{
int f=0;
while(k && t[k]!=v) f=k, k=son[k][v>t[k]];
if(k) ++cnt[k];
else
{
k=++size, sz[k]=cnt[k]=1, fa[k]=f, t[k]=v;
if(f) son[f][v>t[f]]=k;
}
Splay(k,root);
}
void Get_Rank(int k,int v)
{
if(!k) return;
while(son[k][v>t[k]] && t[k]!=v) k=son[k][v>t[k]];
Splay(k,root);
}
int Rank(int k,int x)
{
// ++x;//加了哨兵元素
//if(sz[k]<x) return 0;
while(1)
{
if(sz[son[k][0]]<x && sz[son[k][0]]+cnt[k]>=x) return k;
if(sz[son[k][0]]<x) x-=sz[son[k][0]]+cnt[k],k=son[k][1];
else k=son[k][0];
}
}
int Find(int v,int f)
{
Get_Rank(root,v);
if((t[root]>v && f)||(t[root]<v && !f)) return root;
int k=son[root][f];f^=1;
while(son[k][f]) k=son[k][f];
return k;
}
void Delete(int k,int v)
{
Get_Rank(root,v);
if(cnt[root]>1){--cnt[root],--sz[root];return;}
int x=root;
if(son[root][0] && son[root][1])
{
int k=son[root][1];root=k;
while(son[k][0]) k=son[k][0];
sz[k]+=sz[son[x][0]], fa[son[x][0]]=k, son[k][0]=son[x][0];
Splay(k,root);
}
else
root=son[root][0]+son[root][1];
fa[root]=0;
}
int main()
{
// freopen("testdata.in","r",stdin);
int n=read(),opt,x;
// Insert(root,-INF),Insert(root,INF);
while(n--)
{
opt=read(),x=read();
if(opt==1) Insert(root,x);
else if(opt==2) Delete(root,x);
else if(opt==3) Get_Rank(root,x), printf("%d\n",sz[son[root][0]]+1);
else if(opt==4) printf("%d\n",t[Rank(root,x)]);
else if(opt==5) printf("%d\n",t[Find(x,0)]);
else printf("%d\n",t[Find(x,1)]);
}
return 0;
}
第二次写(2017.12.3):
#include<cstdio>
#include<cctype>
const int N=1e5+5;
int n,root,size,sz[N],cnt[N],t[N],son[N][2],fa[N];
inline int read()
{
int now=0,f=1;register char c=getchar();
for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
for(;isdigit(c);now=now*10+c-'0',c=getchar());
return now*f;
}
inline void Update(int rt)
{
sz[rt]=sz[son[rt][0]]+sz[son[rt][1]]+cnt[rt];
}
void Rotate(int x,int &k)
{
int a=fa[x],b=fa[a],l=son[a][1]==x,r=l^1;
if(a==k) k=x;
else son[b][son[b][1]==a]=x;
fa[x]=b, fa[a]=x, fa[son[x][r]]=a;
son[a][l]=son[x][r], son[x][r]=a;
Update(a),Update(x);
}
void Splay(int x,int &k)
{
while(x!=k)
{
int a=fa[x],b=fa[a];
if(a!=k)
{
if((son[a][0]==x)^(son[b][0]==a)) Rotate(x,k);
else Rotate(a,k);
}
Rotate(x,k);
}
}
void Insert(int v,int k)
{
int f=0;
while(k && t[k]!=v) f=k,k=son[k][v>t[k]];//v>t[k]!
if(k) ++cnt[k];
else
{
k=++size, sz[k]=cnt[k]=1, t[k]=v, fa[k]=f;
if(f) son[f][v>t[f]]=k;
}
Splay(k,root);
}
void Rank(int v,int k)
{
if(!k) return;//!
while(son[k][v>t[k]] && t[k]!=v) k=son[k][v>t[k]];
Splay(k,root);
}
void Delete(int v)
{
Rank(v,root);//!
// while(son[k][v>t[k]] && t[k]!=v) k=son[k][v>t[k]];//!
if(cnt[root]>1) {--cnt[root],--sz[root]; return;}//--sz[k]!
if(son[root][0] && son[root][1])//两种情况!
{
int tmp=root,k=son[root][1];
root=k;//root=k!
while(son[k][0]) k=son[k][0];
sz[k]+=sz[son[tmp][0]],//加sz!
fa[son[tmp][0]]=k, son[k][0]=son[tmp][0];
Splay(k,root);//!
}
else root=son[root][0]+son[root][1];
fa[root]=0;//!
// Splay(k,root);
}
int Get_Rank(int x,int k)
{
while(1)
{
if(sz[son[k][0]]<x && sz[son[k][0]]+cnt[k]>=x) return k;
if(sz[son[k][0]]<x) x-=sz[son[k][0]]+cnt[k],k=son[k][1];
else k=son[k][0];
}
}
int Find(int v,int f)
{
Rank(v,root);
if((t[root]>v && !f)||(t[root]<v && f)) return root;
int k=son[root][f^1];
while(son[k][f]) k=son[k][f];
return k;
}
int main()
{
// freopen("3369.in","r",stdin);
n=read();
int opt,x;
while(n--)
{
opt=read(),x=read();
if(opt==1) Insert(x,root);
else if(opt==2) Delete(x);
else if(opt==3) Rank(x,root),printf("%d\n",sz[son[root][0]]+1);
else if(opt==4) printf("%d\n",t[Get_Rank(x,root)]);
else if(opt==5) printf("%d\n",t[Find(x,1)]);
else printf("%d\n",t[Find(x,0)]);
}
return 0;
}
第三次写(2018.4.4):(码风变化有点大啊)
#include <cstdio>
#include <cctype>
#include <algorithm>
//#define gc() getchar()
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
#define MAXIN 500000
const int N=2e5+5;
char IN[MAXIN],*SS=IN,*TT=IN;
inline int read()
{
int now=0,f=1;register char c=gc();
for(;!isdigit(c);c=gc()) if(c=='-') f=-1;
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now*f;
}
namespace Splay
{
#define lson son[x][0]
#define rson son[x][1]
int root,size,fa[N],son[N][2],sz[N],t[N],cnt[N];
inline void Update(int x){
sz[x]=sz[lson]+sz[rson]+cnt[x];
}
void Rotate(int x,int &k)
{
int a=fa[x],b=fa[a],l=son[a][1]==x,r=l^1;
if(k==a) k=x;
else son[b][son[b][1]==a]=x;
fa[a]=x, fa[x]=b, fa[son[x][r]]=a,
son[a][l]=son[x][r], son[x][r]=a;
Update(a);// Update(x);
}
void Splay(int x,int &k)
{
while(x!=k)
{
int a=fa[x],b=fa[a];
if(a!=k) (son[a][1]==x^son[b][1]==a)?Rotate(x,k):Rotate(a,k);
Rotate(x,k);
}
Update(x);
}
void Insert(int v,int k)
{
int f=0;
while(t[k]!=v&&k) f=k,k=son[k][v>t[k]];
if(k) ++cnt[k];
else{
k=++size, sz[k]=cnt[k]=1, t[k]=v, fa[k]=f;
if(f) son[f][v>t[f]]=k;
}
Splay(k,root);
}
void Rank(int v,int k)
{
// if(!k) return;
while(t[k]!=v && son[k][v>t[k]]) k=son[k][v>t[k]];
Splay(k,root);
}
void Delete(int v,int k)
{
Rank(v,root);
if(cnt[root]>1) --cnt[root],--sz[root];
else if(son[root][0]&&son[root][1])
{
int ls=son[root][0],k=son[root][1];
fa[root=k]=0;
while(son[k][0]) k=son[k][0];
fa[ls]=k, son[k][0]=ls;// sz[k]+=sz[ls];
Splay(k,root);
}
else root=son[root][0]^son[root][1],fa[root]=0;
}
int Get_Rank(int k,int x)
{
while(1)
{
if(sz[lson]<k && sz[lson]+cnt[x]>=k) return x;
if(sz[lson]<k) k-=sz[lson]+cnt[x],x=rson;
else x=lson;
}
}
int Find(int v,int w)
{
Rank(v,root);
if((t[root]>v&&!w)||(t[root]<v&&w)) return root;//!
int x=son[root][w^1];
while(son[x][w]) x=son[x][w];
return x;
}
}
using namespace Splay;
int main()
{
int n=read(),opt;
while(n--)
switch(opt=read())
{
case 1: Insert(read(),root); break;
case 2: Delete(read(),root); break;
case 3: Rank(read(),root),printf("%d\n",sz[son[root][0]]+1); break;
case 4: printf("%d\n",t[Get_Rank(read(),root)]); break;
case 5: printf("%d\n",t[Find(read(),1)]); break;
case 6: printf("%d\n",t[Find(read(),0)]); break;
}
return 0;
}
洛谷.3369.[模板]普通平衡树(Splay)的更多相关文章
- 洛谷.3391.[模板]文艺平衡树(Splay)
题目链接 //注意建树 #include<cstdio> #include<algorithm> const int N=1e5+5; //using std::swap; i ...
- 洛谷.3369.[模板]普通平衡树(fhq Treap)
题目链接 第一次(2017.12.24): #include<cstdio> #include<cctype> #include<algorithm> //#def ...
- 【洛谷P3369】普通平衡树——Splay学习笔记(一)
二叉搜索树(二叉排序树) 概念:一棵树,若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值: 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值: 它的左.右子树也分别为二叉搜索树 ...
- 【洛谷P3391】文艺平衡树——Splay学习笔记(二)
题目链接 Splay基础操作 \(Splay\)上的区间翻转 首先,这里的\(Splay\)维护的是一个序列的顺序,每个结点即为序列中的一个数,序列的顺序即为\(Splay\)的中序遍历 那么如何实现 ...
- 在洛谷3369 Treap模板题 中发现的Splay详解
本题的Splay写法(无指针Splay超详细) 前言 首先来讲...终于调出来了55555...调了整整3天..... 看到大部分大佬都是用指针来实现的Splay.小的只是按照Splay的核心思想和原 ...
- [洛谷日报第62期]Splay简易教程 (转载)
本文发布于洛谷日报,特约作者:tiger0132 原地址 分割线下为copy的内容 [洛谷日报第62期]Splay简易教程 洛谷科技 18-10-0223:31 简介 二叉排序树(Binary Sor ...
- 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)
To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...
- luoguP3391[模板]文艺平衡树(Splay) 题解
链接一下题目:luoguP3391[模板]文艺平衡树(Splay) 平衡树解析 这里的Splay维护的显然不再是权值排序 现在按照的是序列中的编号排序(不过在这道题目里面就是权值诶...) 那么,继续 ...
- LCT总结——概念篇+洛谷P3690[模板]Link Cut Tree(动态树)(LCT,Splay)
为了优化体验(其实是强迫症),蒟蒻把总结拆成了两篇,方便不同学习阶段的Dalao们切换. LCT总结--应用篇戳这里 概念.性质简述 首先介绍一下链剖分的概念(感谢laofu的讲课) 链剖分,是指一类 ...
随机推荐
- Java将文件中的内容转换为sql语句(和并发定时读取文件)
数据文件内容data.txt {USER_TYPE=1,CREATE_USER=ZHANG,UPDATE_USER=li,OPER_NUM=D001,SRC=2,UPDATE_TIME=2018-11 ...
- python 自动获取手机短信验证码
需要一个有权限的 APK 在手机实时存储短信到手机内存 /sdcard/smslog.txt 里(外部SD卡也可以知道能通过adb命令访问到): /***** ...... try {long tim ...
- 如何让谷歌浏览器支持跨域访问(AJAX) AJAX调试跨域接口
以谷歌最新版本为例(2018) 1.在电脑上新建一个目录,例如:C:\MyChromeDevUserData 2.在属性页面中的目标输入框里加上 --disable-web-security -- ...
- 浏览器LocalStroage使用
http://www.cnblogs.com/st-leslie/p/5617130.html
- hbase启动后HMaster进程自动关闭
1.情况描述如题所示,hbase启动以后,HMaster进程启动了,几秒钟以后自动关闭,但是HRegionServer进程正常运行: 原因是,hdfs的默认端口号是8020,而我core-site.x ...
- Spark核心RDD、什么是RDD、RDD的属性、创建RDD、RDD的依赖以及缓存、
1:什么是Spark的RDD??? RDD(Resilient Distributed Dataset)叫做弹性分布式数据集,是Spark中最基本的数据抽象,它代表一个不可变.可分区.里面的元素可并行 ...
- (转载)Memcached和Redis简介
转载自: Memcached和Redis简介 博主的Redis资料列表.http://www.cnblogs.com/programlearning/category/1003158.html 前言: ...
- Java基础知识➣面向对象(八)
概述 Java和C#都是面向对象语言,面向对象编程是目前高级语言习惯的编程模式,与C++编写过程编程而言,面向对象使用起来高效.灵活:面向对象的三个特征:封装.继承和多态. Java面向对象 1.类封 ...
- js获取file控件的完整路径(上传图片预览)
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding= ...
- window与linux文件共享解决方案
我的系统是windows7 x64,虚拟机上的linux系统是centos6.5 方法一: 1.在win7系统上建立一个用户 2.在f盘建立一个文件夹linuxshare,然后右击-属性-共享-高级共 ...