文艺平衡树 Splay 学习笔记(1)

(这里是Splay基础操作,reserve什么的会在下一篇里面讲)

好久之前就说要学Splay了,结果苟到现在才学习。

可能是最近良心发现自己实在太弱了,听数学又听不懂只好多学点不要脑子的数据结构。

感觉Splay比Treap良心多了——代码真的好写。

对于Splay显然可以维护Treap的所有操作,并且本质是BST。

先看看Splay是怎么维护普通平衡树操作的吧。

首先先定义一些基础的变量(若不作特殊说明这些变量的意义不变)

int t[N][2] // t[x][0]表示节点x的左子树,t[x][1]表示节点x的右子树

int cnt[N] // cnt[x]表示节点x存储多少个重复的数

int val[N] // val[x]表示节点x存储数的大小

int par[N] // par[x]表示节点x的直接父亲,特别的,根节点的直接父亲为0

int size[N] // size[x]表示在BST中x子树中存储数的个数

# define ls(x) (t[x][0])

# define rs(x) (t[x][1]) //方便书写

初始化:Splay平衡树中有有+无穷 -无穷两个哨兵节点!!!

Check(x) 函数 :

询问节点x是其父亲的左儿子(return 0)还是右孩子(return 1)

int check (int x)  { return rs(par[x])==x;
}

由上述代码可知,根节点的check(root)值为0,即根节点是0节点的左儿子(Nothing Special)

Up(x)函数:

对于节点x维护其size值为两个孩子的size值+自身cnt值。

void up(int x){size[x]=size[ls(x)]+size[rs(x)]+cnt[x];
}

Rotate(x)函数:

对于节点x旋转到其父亲节点,且不改变树BST性质。(使得树形态较为随机)

这里需要解释一下Rotate的解释和记忆方法(和Treap中Rotate类似)

对于一棵有根树(且父亲指向儿子的边有向),我们现在以把左儿子旋到父节点为例。

第1步,考虑4的右子树已经有元素了,考虑把右子树连接到父节点左儿子处。不改变BST性质。

在此基础上考虑第二步,就是吧2(4的直接父亲接到4的右边)不改变BST性质。

这个时候我们会发现,只进行第三部就可以完成一次rotate操作。

即连一条1指向4的边即可。

对于左边节点转到父亲节点,一般称之为右旋

对于右旋函数的代码,不难得到。

void rotate(int x){int y=par[x];t[y][0]=t[x][1]; par[t[x][1]]=y;t[x][1]=y; t[par[y]][check(y)]=x;par[x]=par[y]; par[y]=x;up(y); up(x);
}
右旋

那左旋呢???

所有0和1的地方去个反不就好了!!!(至少我是那么记的)

对于 真正的旋转代码:

void rotate(int x){int y=par[x],k=check(x);t[y][k]=t[x][k^1]; par[t[x][k^1]]=y;t[x][k^1]=y; t[par[y]][check(y)]=x;par[x]=par[y]; par[y]=x;up(y); up(x);
}

Splay(x,goal) 操作

把节点x通过若干次rotate操作使其到达目标节点goal或者为根,而goal却成为节点x的儿子。

我们可以分三种情况讨论:

1. goal 是 x的直接父亲(边界),那么直接将x旋到父亲位置即可

2. x和x的直接父亲y和x的爷爷z在同一条直线上(不会打结吗?我们需要给定一种顺序)

那么先旋转父亲y,再旋转x,这个时候想就被旋转到y和z节点上方了。

3. x和x的直接父亲y和x的爷爷z在不在同一条直线上(直接把x转两次不就行了么)

我们可以参考下图,模拟一条链上的Splay操作。

 简单的代码实现如下(自然语言是多么的无力....)

void splay(int x,int goal=0) {while (par[x]!=goal) {int y=par[x],z=par[y];if (z!=goal) {if (check(x)==check(y)) rotate(y);else rotate(x);}rotate(x);}if (!goal) root=x;
}

 Insert(x)操作

插入一个值为x的元素。

显然从根节点开始按照BST性质访问Splay,直到找到一个节点v,其值val[v]恰好为x,那么直接增加个数

如果找不到节点的的val[v]恰好为x,那么新建一个节点即可。

最后Splay一下防止出现长链的情况。

void insert(int x) {int cur=root,p=0;while (cur && val[cur]!=x)p=cur,cur=t[cur][x>val[cur]];if (cur) cnt[cur]++;else {cur=++tot;if (p) t[p][x>val[p]]=cur;ls(cur)=rs(cur)=0;par[cur]=p; val[cur]=x;size[cur]=cnt[cur]=1;}splay(cur);
}

Find(x)函数

将val值小于等于x的val值最大一个节点,旋转到根。

题目解决在于如何找到val值小于等于x的val最大的一个节点,注意不能找到空节点,所以要判断。

找到节点v直接利用Splay操作,旋转到根即可。

void find(int x){if (!root) return;int cur=root;while (t[cur][x>val[cur]] && val[cur]!=x)cur=t[cur][x>val[cur]];splay(cur);
}

Rank(x)函数

求值x的排名(x可能曾经没有出现过),排名的定义是比x小的元素个数+1。

利用find操作后,如果x之前出现过,即val[root]=x,那么直接输出左子树的size

否则那么根节点一定比x小那么还需要加上根节点的cnt

int rank(int x){find(x);if (val[root]>=x) return size[ls(root)];else return size[ls(root)]+cnt[root]-1;
}

pre(x)和suc(x)函数

求值x的前驱(比x小的最大数,若没有是-无穷),求值x的后继(比x大的最小数,若没有是+无穷),x可能没有出现过。

求前驱,考虑find操作以后小于等于x的元素都在根及根的左侧,那么如果根直接小于x(x之前没出现过),那么直接打印根就行

否则在左子树中找一直往右子树走找最大的即可。

求后继,考虑find操作以后大于等于x的元素都在根及根的左侧,那么如果根直接大于x(x之前没出现过),那么直接打印根就行

否则在右子树中找一直往左子树走找最小的即可。

int pre_id(int x) {find(x);if (val[root]<x) return root;int cur=ls(root);while (rs(cur)) cur=rs(cur);return cur;
}    
int pre_val(int x){return val[pre_id(x)];
}
int suc_id(int x) {find(x);if (val[root]>x) return root;int cur=rs(root);while (ls(cur)) cur=ls(cur);return cur;
}
int suc_val(int x){return val[suc_id(x)];
}

erase(x)操作

删除权值为x的一个数。

考虑数x的前驱和后继是唯一的,那么求出x的前驱和x的后继,均用Splay操作转到根节点和根节点的右儿子处,

那么根节点右儿子的左儿子一定就可知道是x的了。直接删除它即可,特别的是,剩余个数大于1和等于1的时候需要不同处理

其中大于1的时候,直接吧cnt减去1即可,等于1的时候,则需要删除节点所有的信息。

void erase(int x){int last=pre_id(x),next=suc_id(x);splay(last),splay(next,last);int d=ls(rs(root));if (cnt[d]>1) cnt[d]--,splay(d);else t[next][0]=0;
}

Treap模板题目:https://www.luogu.org/problemnew/show/P3369

# include <bits/stdc++.h>
using namespace std;
const int N=2e5+10;
struct Splay{# define ls(x) (t[x][0])# define rs(x) (t[x][1])# define inf (0x3f3f3f3f)int t[N][2],cnt[N],val[N],size[N],par[N];int root,tot;Splay() { tot=root=0; insert(-inf); insert(inf);}int check(int x) { return rs(par[x])==x; }void up(int x){size[x]=size[ls(x)]+size[rs(x)]+cnt[x];}void rotate(int x){int y=par[x],k=check(x);t[y][k]=t[x][k^1]; par[t[x][k^1]]=y;t[x][k^1]=y; t[par[y]][check(y)]=x;par[x]=par[y]; par[y]=x;up(y); up(x);}void splay(int x,int goal=0) {while (par[x]!=goal) {int y=par[x],z=par[y];if (z!=goal) {if (check(x)==check(y)) rotate(y);else rotate(x);}rotate(x);}if (!goal) root=x;}void insert(int x) {int cur=root,p=0;while (cur && val[cur]!=x)p=cur,cur=t[cur][x>val[cur]];if (cur) cnt[cur]++;else {cur=++tot;if (p) t[p][x>val[p]]=cur;ls(cur)=rs(cur)=0;par[cur]=p; val[cur]=x;size[cur]=cnt[cur]=1;}splay(cur);}void find(int x){if (!root) return;int cur=root;while (t[cur][x>val[cur]] && val[cur]!=x)cur=t[cur][x>val[cur]];splay(cur);}int pre_id(int x) {find(x);if (val[root]<x) return root;int cur=ls(root);while (rs(cur)) cur=rs(cur);return cur;}int pre_val(int x){return val[pre_id(x)];}int suc_id(int x) {find(x);if (val[root]>x) return root;int cur=rs(root);while (ls(cur)) cur=ls(cur);return cur;}int suc_val(int x){return val[suc_id(x)];}int rank(int x){find(x);if (val[root]>=x) return size[ls(root)];else return size[ls(root)]+cnt[root]-1;}int kth_id(int k) {if (k>tot||k<0) return -1;int cur=root;while (true) {if (t[cur][0]&&k<=size[ls(cur)]) cur=ls(cur);else if (k>size[ls(cur)]+cnt[cur]) {k-=size[ls(cur)]+cnt[cur];cur=rs(cur);} else return cur;}}int kth_val(int k){return val[kth_id(k+1)];}void erase(int x){int last=pre_id(x);int next=suc_id(x);splay(last);splay(next,last);int d=ls(rs(root));if (cnt[d]>1) cnt[d]--,splay(d);else t[next][0]=0;}
}tr;
int main()
{int T; scanf("%d",&T);while (T--) {int op,x;scanf("%d%d",&op,&x);switch(op) {case 1:tr.insert(x);break;case 2:tr.erase(x);break;case 3:printf("%d\n",tr.rank(x));break;case 4:printf("%d\n",tr.kth_val(x));break;case 5:printf("%d\n",tr.pre_val(x));break;case 6:printf("%d\n",tr.suc_val(x));break;}}return 0;
}
P3369-Splay做法

 

转载于:https://www.cnblogs.com/ljc20020730/p/10597659.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/249348.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JS使用XMLHttpRequest对象POST收发JSON格式数据

JavaScirpt中的XMLHttpRequest对象提供了对 HTTP 协议的完全访问&#xff0c;使用该对象可以在不刷新页面的情况与服务器交互数据。XMLHttpRequest是实现AJAX技术的关键对象&#xff0c;本站曾整理过一篇介绍该对象的文章&#xff1a; JS使用XMLHttpRequest对象与服务器进行数据…

ShopXO本地化部署安装之centeros 安装Apache2.4.6 + PHP7.0.33 + Mysql5.7.25环境

对于centerOS安装PHP环境&#xff0c;目前网上的帖子都已经比较成熟&#xff0c;具体步骤大家可以自行搜索查看&#xff0c;但是在安装过程中遇到的一些小细节&#xff0c;这些内容往往需要结合多个帖子才能找到答案&#xff0c;在这里简单记录一下。 细节一 如果使用的阿里云…

Spring Boot 扩展点应用之工厂加载机制

Spring 工厂加载机制&#xff0c;即 Spring Factories Loader&#xff0c;核心逻辑是使用 SpringFactoriesLoader 加载由用户实现的类&#xff0c;并配置在约定好的META-INF/spring.factories 路径下&#xff0c;该机制可以为框架上下文动态的增加扩展。 该机制类似于 Java SPI…

Vue.js使用-http请求

Vue.js使用-ajax使用 1.为什么要使用ajax 前面的例子&#xff0c;使用的是本地模拟数据&#xff0c;通过ajax请求服务器数据。 2.使用jquery的ajax库示例 new Vue({el: #app,data: {searchQuery: ,columns: [{name: name, iskey: true}, {name: age},{name: sex, dataSource:…

跨域(Cross-Domain) AJAX for IE8 and IE9

1、有过这样一段代码&#xff0c;是ajax $.ajax({url: "http://127.0.0.1:9001",type: "POST",data: JSON.stringify({"reqMsg":"12345"}),dataType: json,timeout: 1000 * 30,success: function (response) {if(response.n6){dosomet…

移动WEB的页面布局

随着移动互联网的日益普遍&#xff0c;现在移动版本的web应用也应用而生&#xff0c;那么在做移动web页面布局的过程中&#xff0c;应该注意哪些要点呢&#xff1f;现把个人的一些学习经验总结如下&#xff1a; 要点一、piexl 1px 2dp dp dpr dpi ppi 要点二、viewport io…

AnswerOpenCV(1001-1007)一周佳作欣赏

外国不过十一&#xff0c;所以利用十一假期&#xff0c;看看他们都在干什么。一、小白问题http://answers.opencv.org/question/199987/contour-single-blob-with-multiple-object/ Contour Single blob with multiple objectHi to everyone. Im developing an object shape id…

Mysql 开启远程连接

在日常的数据库的使用过程&#xff0c;往往会因为连接权限的问题搞得我们焦头烂额&#xff0c;今天我把我们在数据库连接上的几个误区简单做个记录。内容如下&#xff1a; 误区一&#xff1a;MYSQL密码和数据库密码的区别 mysql密码是我们在安装mysql服务是设置的密码&#xf…

基于jsp+servlet完成的用户注册

思考 &#xff1a; 需要创建实体类吗? 需要创建表吗 |----User 存在、不需要创建了&#xff01;表同理、也不需要了 1.设计dao接口 package cn.javabs.usermanager.dao;import cn.javabs.usermanager.entity.User;/*** 用户的dao接口的设计* author Mryang**/ public interfa…

vue resource then

https://www.cnblogs.com/chenhuichao/p/8308993.html

云开发创建云函数

安装wx-server-sdk时候&#xff0c;终端报错如下&#xff1a; 解决方法&#xff1a; 运行&#xff1a;npm cache clean --force即可 转载于:https://www.cnblogs.com/moguzi12345/p/9758842.html

Java8新特性——函数式接口

目录 一、介绍 二、示例 &#xff08;一&#xff09;Consumer 源码解析 测试示例 &#xff08;二&#xff09;Comparator &#xff08;三&#xff09;Predicate 三、应用 四、总结 一、介绍 FunctionalInterface是一种信息注解类型&#xff0c;用于指明接口类型声明…

CSS3笔记之基础篇(一)边框

效果一、圆角效果 border-radius 实心上半圆&#xff1a; 方法&#xff1a;把高度(height)设为宽度&#xff08;width&#xff09;的一半&#xff0c;并且只设置左上角和右上角的半径与元素的高度一致&#xff08;大于也是可以的&#xff09;。 div {height:50px;/*是width…

JavaSE之Java基础(1)

1、为什么重写equals还要重写hashcode 首先equals与hashcode间的关系是这样的&#xff1a; 1、如果两个对象相同&#xff08;即用equals比较返回true&#xff09;&#xff0c;那么它们的hashCode值一定要相同&#xff1b; 2、如果两个对象的hashCode相同&#xff0c;它们并不一…

bootstarp table

https://www.cnblogs.com/laowangc/p/8875526.html

高级组件——弹出式菜单JPopupMenu

弹出式菜单JPopupMenu&#xff0c;需要用到鼠标事件。MouseListener必须要实现所有接口&#xff0c;MouseAdapter是类&#xff0c;只写你关心的方法&#xff0c;即MouseAdapter实现了MouseListener中的方法 import javax.swing.*; import java.awt.*; import java.awt.event.Mo…

CSS3笔记之基础篇(二)颜色和渐变色彩

效果一、颜色之RGBA RGB是一种色彩标准&#xff0c;是由红(R)、绿(G)、蓝(B)的变化以及相互叠加来得到各式各样的颜色。RGBA是在RGB的基础上增加了控制alpha透明度的参数。 语法&#xff1a; color&#xff1a;rgba(R,G,B,A) 以上R、G、B三个参数&#xff0c;正整数值的取值…

19_03_26校内训练[魔法卡片]

题意 有n张有序的卡片&#xff0c;每张卡片上恰有[1,m]中的每一个数&#xff0c;数字写在正面或反面。每次询问区间[l,r]&#xff0c;你可以将卡片上下颠倒&#xff0c;问区间中数字在卡片上方的并的平方和最大是多少。q,n*m≤1,000,000。 思考 一个很重要的性质&#xff0c;若…

vue 静态图片引入

https://blog.csdn.net/weixin_33862188/article/details/93325502

c:if test=/c:if 使用

1、页面引用<%taglib uri"http://java.sun.com/jsp/jstl/core" prefix"c"%> 2、整形判断&#xff1a; <c:if test"${TEST 1}"> </c:if> 3、判断非空&#xff1a; <c:if test"${empty TEST}"> TEST为空 <…