【笔记】Splay

news2025/1/23 13:44:42

【笔记】Splay 目录

      • 简介
        • 右旋
        • 左旋
      • 核心思想
      • 操作
        • a. Splay
        • b. 插入
        • c. 删除
      • 信息的维护
      • 例题
        • AcWing 2437. Splay
        • P3369 【模板】普通平衡树


简介

Splay 是一种平衡树,并且是一棵二叉搜索树(BST)。

它满足对于任意节点,都有左子树上任意点的值 < 当前节点的值 < 右子树上任意点的值。

优点:支持多种操作。
缺点:常数较大。

单次操作均摊复杂度 O ( log ⁡ n ) O(\log n) O(logn)

(注:关于 Splay 单次操作均摊复杂度的证明见 OI-Wiki)

Splay 基于旋转操作维护树的平衡。旋转分为左旋 (zag) 和右旋 (zig)。

旋转,即在保证平衡树中序遍历不变的前提下,改变整个树的深度。

右旋

zig

对于单次操作 zig(a) ,将节点 a a a 左儿子的左儿子( c c c)接到 a a a 的左儿子处,将 c c c 的右儿子接到 b b b 的左儿子,将 c c c 的右儿子改为 b b b

这样,我们完成了一次右旋操作,操作前后, a a a 左子树的中序遍历都为 DcEbF

左旋

zag

左旋即右旋的逆过程,将每一步反过来即可。


核心思想

每次操作之后,都将被操作的节点旋转至根节点。

这个和均摊时间复杂度有关。

操作

a. Splay

每次调用函数 splay(x, k) 表示把点 x x x 旋转至点 k k k 下方。

特别地,当调用 splay(x, 0) 时,表示把点 x x x 旋转至根节点。

有两种情况:

  1. x , y , z x, y,z x,y,z 成一条链

1

  1. x , y , z x,y,z x,y,z 不成一条链

2

b. 插入
  • 根据 BST 性质,找到该元素所在位置并新建节点。插入后将该节点旋转至根节点。
  • 当要求将一个序列插到 y y y 的后面时:
    1. 找到 y y y 的后继 z z z
    2. y y y 转到根。(splay(y, 0);
    3. z z z 转到 y y y 的下方。(splay(z, y);)由于 z > y z>y z>y,所以 z z z y y y 的左儿子。
    4. 显然,此时将要插入的序列接到 z z z 的左儿子(∅)上即可。

ins

c. 删除
  • 删除一段区间 [ l , r ] [l,r] [l,r]
    1. 分别找到 l l l 的前驱 p p p r r r 的后继 q q q
    2. p p p 转到根节点。
    3. q q q 转到 p p p 下方。由于 p < q p<q p<q,所以 q q q p p p 的左儿子。
    4. 显然,要删除的区间就是点 p p p 的整个左子树。直接变没即可。

信息的维护

以模板题 AcWing 2437. Splay 为例。

本题要求我们进行区间翻转操作。

因此维护两个值:

  1. 以每个点为根节点的子树的大小 size
  2. 区间翻转懒标记 flag

和线段树一样,两个函数 pushuppushdown 分别维护 sizeflag

本题的 Splay 保证中序遍历是当前序列的顺序,不一定满足 BST 性质。


例题

AcWing 2437. Splay

原题链接

本题仅是插入和翻转两个操作。翻转就是把这个区间所在子树的左右儿子分别翻转。

具体细节看代码。

struct Splay_Node
{
    int s[2], p; // 左右儿子、父节点
    int v, size, flag; // 值、子树大小、懒标
    
    void init(int _v, int _p) // 初始化
    {
        v = _v, p = _p;
        size = 1;
    }
}tr[N];

int n, m;
int root, idx;

void pushup(int u) // 更新当前节点大小
{
    tr[u].size = tr[tr[u].s[0]].size + tr[tr[u].s[1]].size + 1;
}

void pushdown(int u) // 将懒标记下传
{
    if (tr[u].flag) // 如果当前节点有懒标
    {
        swap(tr[u].s[0], tr[u].s[1]); // 就交换左右儿子
        tr[tr[u].s[0]].flag ^= 1; // 左儿子懒标记更新
        tr[tr[u].s[1]].flag ^= 1; // 右儿子懒标记更新
        tr[u].flag ^= 1; // 当前节点懒标记清空
    }
}

void rotate(int x) // 旋转
{
    int y = tr[x].p, z = tr[y].p; // 当前节点的父亲和祖父
    int k = tr[y].s[1] == x, kk = tr[z].s[1] == y; // 0 -> left | 1 -> right
    // 这里一个小技巧判断哪个儿子
    tr[z].s[kk] = x, tr[x].p = z; // z的儿子改为x
    tr[y].s[k] = tr[x].s[k ^ 1], tr[tr[x].s[k ^ 1]].p = y; // y的儿子改为x的反儿子
    tr[x].s[k ^ 1] = y, tr[y].p = x; // x的反儿子变成y
    pushup(y), pushup(x); // 更新节点x,y
}

void splay(int x, int k)
{
    while (tr[x].p != k) // 如果k不是当前节点的父节点
    {
        int y = tr[x].p, z = tr[y].p; // 当前节点的父亲和祖父
        if (z != k) // 如果k不是当前节点的祖父
            if ((tr[y].s[1] == x) ^ (tr[z].s[1] == y)) rotate(x); // 不成链先转x
            else rotate(y); // 成链先转y
        rotate(x); // 最后都转一遍x
    }
    if (!k) root = x; // 如果当前x是根结点就更新root
}

void insert(int v) // 插入
{
    int u = root, p = 0; // 当前节点和其父节点编号
    while (u) p = u, u = tr[u].s[v > tr[u].v]; // 只要节点存在就往下找
    // 后面那句意思是如果插入的值比当前节点小就去左子树,否则去右子树
    u = ++ idx; // 动态开点编号
    if (p) tr[p].s[v > tr[p].v] = u; // 如果u不是根结点就更新p的儿子为u
    tr[u].init(v, p); // 初始化新的点u
    splay(u, 0); // 将u整到根结点
}

int kth(int k) // 找第k小数
{
    int u = root; // 从根结点开始找
    while (tr[u].size >= k)
    {
        pushdown(u); // 找之前先下传懒标记
        if (tr[tr[u].s[0]].size >= k) u = tr[u].s[0]; // 如果k比左子树小的话就去左子树
        else if (tr[tr[u].s[0]].size + 1 == k) return splay(u, 0), u; // 如果刚好在当前点就返回
        else k -= tr[tr[u].s[0]].size + 1, u = tr[u].s[1]; // 否则去右子树
    }
    return -1; // 找不到就返回-1
}

void output(int u) // 输出中序遍历"左-根-右"
{
    pushdown(u); // 访问之前先下传
    if (tr[u].s[0]) output(tr[u].s[0]); // 如果左子树存在就遍历左子树
    if (tr[u].v >= 1 && tr[u].v <= n) printf("%d ", tr[u].v); // 根结点不是哨兵就输出根结点
    if (tr[u].s[1]) output(tr[u].s[1]); // 如果右子树存在就遍历右子树
}

int main()
{
    scanf("%d%d", &n, &m);
    for (int i = 0; i <= n + 1; i ++ ) // 多加2哨兵防止越界
        insert(i);
    
    int l, r;
    while (m -- )
    {
        scanf("%d%d", &l, &r);
        l = kth(l), r = kth(r + 2); // 由于前面加了一个哨兵所以如果我们想要提取区间[l,r]就要以l和r+2分割
        splay(l, 0), splay(r, l); // 将l转到根节点,将r+2转到根节点下方
        tr[tr[r].s[0]].flag ^= 1; // 把r+2的左子树打上懒标记
    }
    
    output(root);
    
    return 0;
}

P3369 【模板】普通平衡树

原题链接

您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:

  1. 插入 x x x
  2. 删除 x x x 数(若有多个相同的数,应只删除一个)
  3. 查询 x x x 数的排名(排名定义为比当前数小的数的个数 + 1 +1 +1 )
  4. 查询排名为 x x x 的数
  5. x x x 的前驱(前驱定义为小于 x x x,且最大的数)
  6. x x x 的后继(后继定义为大于 x x x,且最小的数)
  • 插入:根据 BST 的性质,找到这个值所在的节点。如果该节点存在,则将 cnt + 1 \text{cnt}+1 cnt+1。如果不存在就新建一个节点。
  • 删除:找到这个值的前驱 prev \text{prev} prev 和后继 next \text{next} next(节点编号),将 prev \text{prev} prev 转到根节点,将 next \text{next} next 转到 prev \text{prev} prev 下方。如果 next \text{next} next 左儿子 cnt > 1 \text{cnt}>1 cnt>1 则将 cnt − 1 \text{cnt}-1 cnt1,否则直接删除左儿子。
  • 根据数值找排名:将该数值对应的节点转到根节点,然后返回左子树的大小 + 1 +1 +1
  • 根据排名找数值:从根结点开始找,如果 k k k 比左子树小的话就去左子树,如果刚好在当前点就把这个点转上去并返回,否则去右子树。
  • 求前驱:根据 BST 性质先找出它的位置转到根节点。如果这个值不存在即根节点值小于输入值,则返回根节点值。否则返回根结点左子树的最右儿子。
  • 求后继:根据 BST 性质先找出它的位置转到根节点。如果这个值不存在即根节点值大于输入值,则返回根节点值。否则返回根结点右子树的最左儿子。
struct Node
{
    int size, cnt, v;
    int p, s[2];
    
    void init(int _v, int _p)
    {
        v = _v, p = _p;
        size = 1;
    }
}tr[N];

int n;
int root, idx;

void pushup(int x) // 更新子树大小
{
    tr[x].size = tr[tr[x].s[0]].size + tr[tr[x].s[1]].size + tr[x].cnt; // 注意因为值可以重复,所以加cnt
}

void rotate(int x) // 旋转
{
    int y = tr[x].p, z = tr[y].p;
    int k = tr[y].s[1] == x;
    tr[z].s[tr[z].s[1] == y] = x, tr[x].p = z;
    tr[y].s[k] = tr[x].s[k ^ 1], tr[tr[x].s[k ^ 1]].p = y;
    tr[x].s[k ^ 1] = y, tr[y].p = x;
    pushup(y), pushup(x);
}

void splay(int x, int k) // 这个可以去翻上面的注释
{
    while (tr[x].p != k)
    {
        int y = tr[x].p, z = tr[y].p;
        if (z != k)
            if ((tr[y].s[1] == x) ^ (tr[z].s[1] == y)) rotate(x);
            else rotate(y);
        rotate(x);
    }
    if (!k) root = x;
}

void upper(int v) // 根据BST性质找值v所在的节点并转到根节点
{
    int u = root; // 从根结点开始
    while (tr[u].s[v > tr[u].v] && tr[u].v != v) // 如果值比当前节点小就去左子树,否则去右子树
        u = tr[u].s[v > tr[u].v];
    splay(u, 0); // 找到之后将这个节点转到根节点
    // 如果这个值不存在,则显然会返回这个值前驱或后继所在的节点
}

int get_prev(int v) // 找前驱
{
    upper(v); // 转到根节点
    if (tr[root].v < v) return root; // 如果这个值不存在且根结点值小则根节点就是前驱
    int u = tr[root].s[0]; // 从左子树开始搜
    while (tr[u].s[1]) u = tr[u].s[1]; // 左子树最右面
    return u;
}

int get_next(int v) // 找后继
{
    upper(v); // 转到根节点
    if (tr[root].v > v) return root; // 如果这个值不存在且根结点值大则根结点就是后继
    int u = tr[root].s[1]; // 从右子树开始搜
    while (tr[u].s[0]) u = tr[u].s[0]; // 右子树最左面
    return u;
}

int get_rank_by_val(int v) // 根据数值找排名
{
    upper(v); // 转到根节点
    return tr[tr[root].s[0]].size + 1; // 左子树大小+1
}

int get_val_by_rank(int k) // 根据排名找数值
{
    int u = root;
    while (tr[u].size >= k)
    {
        if (tr[tr[u].s[0]].size >= k) u = tr[u].s[0];
        else if (tr[tr[u].s[0]].size + tr[u].cnt >= k) return splay(u, 0), tr[u].v; // 记得把当前点转上去
        else k -= tr[tr[u].s[0]].size + tr[u].cnt, u = tr[u].s[1];
    }
    return -1;
}

void insert(int v) // 插入一个值
{
    int u = root, p = 0;
    while (u && tr[u].v != v) p = u, u = tr[u].s[v > tr[u].v];
    if (u) tr[u].cnt ++ ; // 如果这个点已经存在就把cnt+1
    else
    {
        u = ++ idx; // 否则新建一个点
        if (p) tr[p].s[v > tr[p].v] = u; // 如果新建的不是根结点就更新其父节点的儿子指针
        tr[u] = {1, 1, v, p}; // 初始化
    }
    splay(u, 0);
}

void remove(int v) // 移除一个值
{
    int prev = get_prev(v), next = get_next(v); // 找出前驱和后继
    splay(prev, 0), splay(next, prev); // 将前驱转到根节点,将后继转到前驱下方
    int w = tr[next].s[0]; // 后继的左儿子是要删除的值
    if (tr[w].cnt > 1) tr[w].cnt -- , splay(w, 0); // 如果不止一个就把cnt-1然后转上去
    else tr[next].s[0] = 0, splay(next, 0); // 否则把next左儿子指针置空然后把后继转上去
}

void output(int u)
{
    if (tr[u].s[0]) output(tr[u].s[0]);
    if (tr[u].v != -INF && tr[u].v != INF) printf("%d ", tr[u].v);
    if (tr[u].s[1]) output(tr[u].s[1]);
}

int main()
{
    int op, x;
    insert(INF), insert(-INF); // 为防止出界整两个哨兵
    
    scanf("%d", &n);
    while (n -- )
    {
        scanf("%d%d", &op, &x);
        switch (op)
        {
            case 1: insert(x); break;
            case 2: remove(x); break;
            case 3: printf("%d\n", get_rank_by_val(x) - 1); break; // 由于有哨兵,所以排名-1
            case 4: printf("%d\n", get_val_by_rank(x + 1)); break; // 由于有哨兵,所以输入+1
            case 5: printf("%d\n", tr[get_prev(x)].v); break; // 由于找前驱返回的是下标
            case 6: printf("%d\n", tr[get_next(x)].v); break; // 所以输出数值
        }
    }
    
    return 0;
}

最后,如果觉得对您有帮助的话,点个赞再走吧!

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

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

相关文章

fdbus之CBaseMessage

总体介绍 这个类是一个很重要的的类&#xff0c;fdbus中传递对象就是这个类的实例&#xff0c;该类中包含了很多重要的信息。可以这样理解&#xff0c;再fdbus的通信中&#xff0c;这个类的地位至关重要。他们的通信的内容就是该类定义的一些信息。 虽然CFdbBaseObject定义了…

如何进一步全面提高项目估算精准度?

项目估算非常重要&#xff0c;这直接关系着项目的成本和收入&#xff0c;如果估算不准确&#xff0c;将为项目带来较大风险。一般软件规模可以用多种方式进行估算&#xff0c;但是用功能点估算方式更准确&#xff0c;而自动估算让估算更快速&#xff0c;我们以CoCode开发的估算…

初识网络编程

一、概述 地球村&#xff1a;亦称世界村&#xff0c;是通过电子媒介将世界紧密联系起来的形象表达&#xff0c;是信息网络时代的集中体现 TCP和UDP&#xff1a; TCP&#xff1a;打电话 -->连接 -->接了 -->通话 UDP&#xff1a;发送完即可 -->接收 计算机网络&a…

QQ表情包存储位置解析

一些常见的设备和系统的QQ表情包存储位置&#xff1a; Windows系统&#xff1a; 路径&#xff1a;C:\Users[用户名]\Documents\Tencent Files[QQ号码]\Image\Image\CustomFace 在这个文件夹中&#xff0c;您可以找到所有自定义的QQ表情包。 Android系统&#xff1a; 路径&am…

程序开发常用在线工具汇总

菜鸟工具# https://c.runoob.com/ 编码# ASCII码# https://www.habaijian.com/ 在线转换# https://www.107000.com/T-Ascii/http://www.ab126.com/goju/1711.html Base64# 在线转换# https://www.qqxiuzi.cn/bianma/base64.htmhttp://www.mxcz.net/tools/Unicode.aspx …

软件架构的演化和维护

软件架构的演化和维护 定义 定义 顶不住了&#xff0c;刷题去了&#xff0c;不搞这个了&#xff0c;想吐。。。

STM32Cube 开发之读写内部Flash--电源项目ADC采样校准系数存储-实现掉电读取数据--STM32或者GD32F处理器

STM32Cube 开发之读写内部Flash–电源项目ADC采样校准系数存储-实现掉电读取数据 一、需求介绍 1.1 在进行电源项目开发中&#xff0c;输入与输出的电压电流经过硬件电路分压或者差分变换后&#xff0c;将低压的电压信号给到单片机如STM32F1系列单片机的ADC采样端口&#xff…

网速Full Power!这款4G网关信号达360度无死角

数字化转型浪潮下,如何实现可靠的无线互联成为制造企业面临的新课题。广州数智自动化最近通过部署星创SG500 4G网关,成功实现了某工业园区全域无线覆盖和多系统安全访问。 SG500支持全球主流的4G网络频段,可灵活搭配通信运营商,提供高达150Mbps的无线传输速率。它采用强大的四核…

性能压测工具:wrk

一般我们压测的时候&#xff0c;需要了解衡量系统性能的一些参数指标&#xff0c;比如。 1、性能指标简介 1.1 延迟 简单易懂。green:一般指响应时间 95线&#xff1a;P95。平均100%的请求中95%已经响应的时间 99线&#xff1a;P99。平均100%的请求中99%已经响应的时间 平…

三.vue2路由知识全总结

Vue Devtools&#xff1a;插件安装&#xff0c;展示模块中的数据 vue-router 应用场景&#xff1a;Vue Router 是 Vue.js 的官方路由。它与 Vue.js 核心深度集成&#xff0c;让用 Vue.js 构建单页应用变得轻而易举。 嵌套的路由/视图表模块化的、基于组件的路由配置路由参数、…

谷器数据参加世界制造业大会及数字化转型高峰论坛

9月20日至24日&#xff0c;由工业和信息化部、科技部、商务部、国务院国资委、中国工程院、安徽省人民政府等单位组织共同主办的2023世界制造业大会在合肥市滨湖国际会展中心盛大举行。谷器数据受邀出席&#xff0c;并同期参加”数字化转型高峰论坛”&#xff0c;与国家工信部相…

向量数据库X云计算驱动大模型落地电商行业,Zilliz联合AWS探索并贡献成熟解决方案

近日,由Zilliz 联合亚马逊云科技举办的【向量数据库 X 云计算 驱动大模型落地电商行业】活动在上海落幕,获得业内专业人士的广泛好评。 众所周知,大模型技术的发展正加速对千行万业的改革和重塑,向量数据库作为大模型的海量记忆体、云计算作为大模型的大算力平台,是大模型…

机器人中的数值优化|【五】BFGS算法非凸/非光滑处理

机器人中的数值优化|【五】BFGS算法的非凸/非光滑处理 往期内容回顾 机器人中的数值优化|【一】数值优化基础 机器人中的数值优化|【二】最速下降法&#xff0c;可行牛顿法的python实现&#xff0c;以Rosenbrock function为例 机器人中的数值优化|【三】无约束优化&#xff0…

微信里怎么添加阅读付费链接

在微信中添加阅读付费链接为主题&#xff0c;首先需要开通微信支付商户号&#xff0c;然后创建自定义菜单&#xff0c;并设置跳转到付费链接的逻辑。以下是详细步骤&#xff1a; 注册并开通微信支付商户号 在微信开放平台上注册并开通微信支付商户号。这一步需要营业执照、法…

企业如何识别和满足客户需求的5个要点

随着市场竞争的日益加剧&#xff0c;企业需要更加注重客户需求&#xff0c;以获得持续的发展。而企业在满足客户需求上&#xff0c;则需要遵循一些基本的原则和方法。本文将介绍企业识别和满足客户需求的5个要点。 1、理解客户 企业需要了解客户的需求、想法和行为&#xff0c…

Python 3.12.0 正式版即将发布!

导读Python 3.12.0 发布了第 2 个 RC 版本&#xff0c;也是最后一个 RC。正式版将于 2023 年 10 月 2 日星期一发布。 开发团队表示&#xff0c;进入候选版本阶段后&#xff0c;只接受经过 review 且修复明确错误的代码。RC2 是发现并修复重要问题的最后机会。 从该版本开始&a…

安装nvm包含卸载node及卸载nvm

卸载node 1、在命令行输入where node查看node所在位置&#xff0c;删除node.exe所在的父级文件夹 2、控制面板中程序卸载&#xff0c;卸载node.js 安装nvm 1、安装nvm-setup.exe 2、命令行运行nvm v&#xff0c;如果出现版本号表示安装成功 3、从node官网下载node版本&#…

事件循环——message loop

1 浏览器的进程模型 1.1 进程 程序运行需要有它自己专属的内存空间&#xff0c;可以把这块内存空间简单的理解为进程。 每个应用至少有一个进程&#xff0c;进程之前相互独立&#xff0c;即使要通信&#xff0c;也需要双方同意。比如&#xff1a;qq、微信、王者荣耀进程。 …

电视盒子什么牌子好?花费30天测评盘点超值电视盒子推荐

最近超多网友咨询我不知道电视盒子什么牌子好&#xff0c;为了推荐结果更客观公正&#xff0c;我将最热门的十款电视盒子买回来进行了一个月的深度测评&#xff0c;最终筛选了五款最优秀的电视盒子推荐给大家&#xff0c;不懂电视盒子怎么挑选那这篇文章就不能错过了。 1、泰捷…

3+单基因泛癌+铜死亡纯生信思路

今天给同学们分享一篇3单基因泛癌铜死亡纯生信思路的生信文章“Systematic pan-cancer analysis identifies SLC31A1 as a biomarker in multiple tumor types”&#xff0c;这篇文章于2023年3月27日发表在BMC Med Genomics 期刊上&#xff0c;影响因子为3.622。 溶质载体家族3…