皆さん、こんにちは。上一篇文章,我们讲了Splay如何实现。这一篇我们来让我们的伸展树短一点。

上一篇Splay讲解的链接:リンク

首先还是变量的定义,在这里呢,我把一些小函数也用Define来实现了。

1).变量定义和小函数。

#include<cstdio>
using namespace std;
using namespace std;
const int MAXN=1e5+5;
const int INF=1<<30;
struct Floor{ int Val,Fa,Size,Recy,Ch[2];}T[MAXN];//定义伸展树的结构体。
int N,Opt,X,NodeNum,TSize;//一些基础变量。
//定义根为零结点的右儿子。
#define Root RS(0)
//用Define来表示各各变量。
#define Val(X) T[X].Val
#define LS(X) T[X].Ch[0]
#define RS(X) T[X].Ch[1]
#define Fa(X) T[X].Fa
#define Recy(X) T[X].Recy
#define Size(X) T[X].Size
//更新函数
#define Update(X) Size(X)=Size(LS(X))+Size(RS(X))+Recy(X)
//关系函数
#define Ident(X) (RS(Fa(X))==X)
//连接函数
#define Connect(X,Y,Son) T[Fa(X)=Y].Ch[Son]=X

2).旋转函数

void Rotate(int X){
int YSon=Ident(X),Y=Fa(X),RSon=Ident(Y),R=Fa(Y),XSon=YSon^1,B=T[X].Ch[XSon];//记录位置和关系。
Connect(B,Y,YSon);Connect(X,R,RSon);Connect(Y,X,XSon);//旋转三板斧。
Update(Y);Update(X);//更新。
}

3).伸展函数

void Splay(int From,int To){
To=Fa(To);
while(Fa(From)!=To){
int Up=Fa(From);
if(Fa(Up)!=To) Ident(From)^Ident(To) ? Rotate(From):Rotate(Up);//判断是“之”字形还是“一”字型。
Rotate(From);
}
}

4).查找函数

int Find(int Value){
int Now=Root;
while(Now){
if(Value==Val(Now)){ Splay(Now,Root);return Now;}
Now=T[Now].Ch[Value>Val(Now)];//判断Now向左还是向右。
}
return 0;
}

5).创造结点

void CreNode(int Value,int Father){
NodeNum++;
Val(NodeNum)=Value;
Fa(NodeNum)=Father;
Size(NodeNum)=Recy(NodeNum)=1;
}

6).创造结点

int Insert(int Value){
if(!TSize){
TSize++;
Root=NodeNum+1;
CreNode(Value,0);
return NodeNum;
}
TSize++;
int Now=Root;
while(Now){
Size(Now)++;
if(Val(Now)==Value){ Recy(Now)++;return Now;}
int Next=Value<T[Now].Val ? 0:1;
if(!T[Now].Ch[Next]){
T[Now].Ch[Next]=NodeNum+1;
CreNode(Value,Now);
return NodeNum;
}
Now=T[Now].Ch[Next];
}
return 0;
}

7).创造结点

void Push(int Value){ int AddNode=Insert(Value);Splay(AddNode,Root);}

8).删除结点

void Destroy(int X){ Val(X)=Fa(X)=Size(X)=Recy(X)=LS(X)=RS(X)=0;}

9).删除结点

void Pop(int Value){
int KillNode=Find(Value);
if(!KillNode) return;
TSize--;
if(Recy(KillNode)>1){ Recy(KillNode)--;Size(KillNode)--;return;}
else if(!LS(KillNode)){ Root=RS(KillNode);Fa(Root)=0;}
else{
int LMAX=LS(KillNode);
while(RS(LMAX)) LMAX=RS(LMAX);
int R=RS(KillNode);
Splay(LMAX,LS(KillNode));
Connect(R,LMAX,1);Connect(LMAX,0,1);
Update(LMAX);
}
Destroy(KillNode);
}

10).查找值为X的排名

int Rank(int Value){
int Now=Root,Ans=1;
while(Now){
if(Value==Val(Now)){
Ans+=Size(LS(Now));
Splay(Now,Root);
return Ans;
}else{
Ans=Ans+(Value>Val(Now))*(Size(LS(Now))+Recy(Now));
Now=T[Now].Ch[Value>Val(Now)];
}
}
return 0;
}

11).查找第X小的数

int Atrank(int X){
if(X>TSize) return 0;
int Now=Root;
while(Now){
if(Size(LS(Now))<X && X<=Size(Now)-Size(RS(Now))){
Splay(Now,Root);
return Val(Now);
}else if(X<=Size(LS(Now))) Now=LS(Now);
else{
X=X-(Size(Now)-Size(RS(Now)));
Now=RS(Now);
}
}
return 0;
}

12).查找值为X的前驱

int Lower(int Value){
int Now=Root,Ans=-INF;
while(Now){
if(Value>Val(Now) && Val(Now)>Ans) Ans=Val(Now);
Now=T[Now].Ch[Value>Val(Now)];
}
return Ans;
}

13).查找值为X的后继

int Upper(int Value){
int Now=Root,Ans=INF;
while(Now){
if(Value<Val(Now) && Val(Now)<Ans) Ans=Val(Now);
Now=T[Now].Ch[Value>=Val(Now)];
}
return Ans;
}

14).主函数

int main(){
scanf("%d",&N);
while(N--){
scanf("%d%d",&Opt,&X);
if(Opt==1) Push(X);
else if(Opt==2) Pop(X);
else if(Opt==3) printf("%d\n",Rank(X));
else if(Opt==4) printf("%d\n",Atrank(X));
else if(Opt==5) printf("%d\n",Lower(X));
else printf("%d\n",Upper(X));
}
return 0;
}

再悄悄说一声,把上面所有的代码拼起来也可以AC平衡树模板题。

Splay代码简化版的更多相关文章

  1. 及其简短的Splay代码

    #include <stdio.h> #include <queue> #include <algorithm> #include <stdlib.h> ...

  2. 普通平衡树与文艺平衡树的splay代码

    主要综合借鉴了yyb和马前卒两位大佬的. //普通平衡树 #include <cstdio> #include <cctype> #include <cstring> ...

  3. java:struts2.3框架1(struts2快速配置,各文件之间的关系,基础代码简化版,XML中的通配符)

    1.struts2快速配置: A.到http://struts.apache.org下载struts2开发包struts-2.3.32-all.zip B.新建web项目并添加struts2依赖的ja ...

  4. 算法笔记--Splay && Link-Cut-Tree

    Splay 参考:https://tiger0132.blog.luogu.org/slay-notes 普通模板: ; ], val[N], cnt[N], fa[N], sz[N], lazy[N ...

  5. 【Luogu】P1486郁闷的出纳员(Splay)

    题目链接 名副其实的调了一下午…… 每做一道题都是对我那不规范的Splay代码的刀刻斧凿一般的修正啊…… Splay.如果有一批员工不干了,那就找还能干的薪水最少的员工,把它splay到根,删除它的左 ...

  6. codevs 1743 反转卡片 rope or splay

    [codevs1743]反转卡片 题目描述 Description [dzy493941464|yywyzdzr原创] 小A将N张卡片整齐地排成一排,其中每张卡片上写了1~N的一个整数,每张卡片上的数 ...

  7. SPOJ - QMAX3VN (4350) splay

    SPOJ - QMAX3VN 一个动态的序列 ,在线询问某个区间的最大值.关于静态序列的区间最值问题,用ST表解决,参考POJ 3264 乍一看上去 splay可以轻松解决.书上说可以用块状链表解决, ...

  8. Bzoj1208 [HNOI2004]宠物收养所

    Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 7457  Solved: 2960 Description 最近,阿Q开了一间宠物收养所.收养所提供两 ...

  9. NOI2004 郁闷的出纳员

    Description OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作,但是令人郁闷的是,我们的老板反复无常,经常 ...

随机推荐

  1. Oracle创建包

    包: 在公司中,如果业务逻辑比较复杂,需要定义很多过程或者函数.有可能需要定义几十个过程或者函数,这些过程或者函数如果都放到一起,是不是不好管理?我们一般使用包来管理过程或者函数,一个包中可以定义多个 ...

  2. spring学习笔记(四)我对spring中bean生命周期的理解

    我相信大部分同学对spring中bean的生命周期都不陌生,但是如果要详细的说出每一个步骤,可能能说出来的也不多,我之前也是这样,前几天调了一下spring的源码,看了一点书,突然一下明朗了,理解了s ...

  3. 近期总结的一些Java基础

    1.面向过程:当需要实现一个功能的时候,每一个过程中的详细步骤和细节都要亲力亲为. 2.面向对象:当需要实现一个功能的时候,不关心详细的步骤细节,而是找人帮我做事. 3.类和对象的关系:   a-类是 ...

  4. Java网络小结

    1,定位 IP对机器的定位 端口对软件的定位(65535) URL对软件上每一份资源的定位 2,TCP和UDP TCP 安全,性能低 ①ServerSocket②Socket UDP不安全,性能高 ① ...

  5. P2220 [HAOI2012]容易题(快速幂)

    Describe 为了使得大家高兴,小Q特意出个自认为的简单题(easy)来满足大家,这道简单题是描述如下: 有一个数列A已知对于所有的A[i]都是1~n的自然数,并且知道对于一些A[i]不能取哪些值 ...

  6. python--集合和文件基本操作

    集合 # 集合天生就能去重,集合也是无序的  集合也是{ }  但是空集合定义特殊s=set()  #空集合s2 = set('1234445566778')print(s2) s3 = {'1',' ...

  7. Java—线程的生命周期及线程控制方法详解

    线程生命周期5种状态 介绍   线程的生命周期经过新建(New).就绪(Runnable).运行(Running).阻塞(Bolocked)和死亡(Dead) 状态转换图 新建(New)   程序使用 ...

  8. Java开发架构篇:DDD模型领域层决策规则树服务设计

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 在上一章节介绍了领域驱动设计的基本概念以及按照领域驱动设计的思想进行代码分层,但是仅 ...

  9. 【雕爷学编程】Arduino动手做(60)---WS2812直条8位模块

    37款传感器与执行器的提法,在网络上广泛流传,其实Arduino能够兼容的传感器模块肯定是不止这37种的.鉴于本人手头积累了一些传感器和执行器模块,依照实践出真知(一定要动手做)的理念,以学习和交流为 ...

  10. JS 把数字转换成字母

     JS 把数字转换成字母 2013-03-12 22:28:11 分类: JavaScript String.fromCharCode(addcount+65) 位运算alert(1<<0 ...