【数据结构】Treap树堆

news2024/11/24 13:28:17

Treap树堆

Treap是一种平衡化二叉搜索树,在键值key满足二叉搜索树的前提下,增加了priority是满足堆序的条件。可以证明,如果priority的随机的,那么Treap的期望深度是 O ( l o g N ) O(logN) O(logN),也就是说大部分操作可以在 O ( l o g N ) O(logN) O(logN)的时间内完成。

基本数据定义

Treap就是Tree+heap,首先有二叉搜索树(BST)的数据,treapCnt记录当前的节点总数,节点和键值的映射key,每个节点的左右儿子childs,以及一个键可能存在多个副本,用cnt记录副本数,size记录子树大小,用于查找第k大元素。堆相关的数据:每个节点的优先级priority,这个优先级在创建节点时随机生成,保证了Treap的深度不会太大。

int root, treapCnt, key[maxNode], priority[maxNode];
int childs[maxNode][2], cnt[maxNode], size[maxNode];

旋转操作

Treap的核心操作,通过旋转维护Treap堆的性质(父亲节点的键值大于它所有的儿子节点的)同时不会破坏BST的性质(父亲节点的键值大于左儿子的,大于右儿子的)。

在这里插入图片描述

void rotate(int &x, int t) {
    // t=0/1对应左/右旋操作
    int y = childs[x][t];
    childs[x][t] = childs[y][1 - t];
    childs[y][1 - t] = x;
    // 旋转只会改变自己以及旋转上来的儿子节点
    update(x);
    update(y);
    x = y;
}

插入操作

插入操作在BST的插入基础上实现,插入节点递归返回的过程中,需要通过旋转“修复“堆的性质。

void _insert(int &x, int k) {
    if (x) {
        if (key[x] == k) {
            cnt[x]++;
        }   // 若节点已经存在,则副本数++
        else {
            int t = (key[x] < k);   // 根据BST性质进行插入
            _insert(childs[x][t], k);
            // 完成插入之后,通过旋转保持堆的性质
            if (priority[childs[x][t]] < priority[x]) {
                rotate(x, t);
            }
        }
    }
    else {  // 创建新节点
        x = treapCnt++;
        key[x] = k;
        cnt[x] = 1;
        priority[x] = rand();   // 随机优先级
        childs[x][0] = childs[x][1] = 0;
    }
    update(x);
}

删除操作

同样在BST删除操作的基础上,找到该目标节点之后,不可以直接删除之,需要将优先级更高的儿子节点旋转上来,作为新的父节点,其实可以理解为将待删除节点旋转至叶子节点再进行删除。
有一点要注意,删除一个节点之后,其编号将不再使用,下一个新创建节点的编号为treapCnt,因此在设置数组空间时,也就是确定maxNode时需要考虑极端情况,所有操作均为插入操作。

void _erase(int &x, int k) {
    if (key[x] == k) {
        if (cnt[x] > 1) {
            cnt[x]--;
        }   // 若节点不只1个副本,则副本数--
        else {
            if (childs[x][0] == 0 && childs[x][1] == 0) {
                x = 0;
                return;
            }   // 若没有儿子节点直接删除并返回
            // 否则需要将儿子旋转上来,再递归地进行删除
            int t = (priority[childs[x][0]] > priority[childs[x][1]]);
            rotate(x, t);
            _erase(x, k);
        }
    }
    else {  // 根据BST性质查找目标值
        _erase(childs[x][key[x] < k], k);
    }
    update(x);
}

luogu P3369

查询数x的排名,求数x的前驱/后驱,这3个操作根据BST的性质操作即可。

#include <bits/stdc++.h>
using namespace std;
using ll = long long;

const int maxNode = 1e6 + 5;
struct Treap {
    int root, treapCnt, key[maxNode], priority[maxNode];
    int childs[maxNode][2], cnt[maxNode], size[maxNode];
    // treapCnt:树堆当前节点总数
    // 节点的基本属性:key:键值,priority:随机得到的优先级
    // childs:[i][0]代表左儿子,[i][1]代表右儿子
    // size: 维护子树大小,用于查询排名
    // cnt:每个节点包含的副本数
    Treap() {
        root = 0;
        treapCnt = 1;
        priority[0] = INT_MAX;
        size[0] = 0;
    }

    void update(int x) {
        size[x] = size[childs[x][0]] + cnt[x] + size[childs[x][1]];
    }

    void rotate(int &x, int t) {
        // t=0/1对应左/右旋操作
        int y = childs[x][t];
        childs[x][t] = childs[y][1 - t];
        childs[y][1 - t] = x;
        // 旋转只会改变自己以及旋转上来的儿子节点
        update(x);
        update(y);
        x = y;
    }

    void _insert(int &x, int k) {
        if (x) {
            if (key[x] == k) {
                cnt[x]++;
            }   // 若节点已经存在,则副本数++
            else {
                int t = (key[x] < k);   // 根据BST性质进行插入
                _insert(childs[x][t], k);
                // 完成插入之后,通过旋转保持堆的性质
                if (priority[childs[x][t]] < priority[x]) {
                    rotate(x, t);
                }
            }
        }
        else {  // 创建新节点
            x = treapCnt++;
            key[x] = k;
            cnt[x] = 1;
            priority[x] = rand();   // 随机优先级
            childs[x][0] = childs[x][1] = 0;
        }
        update(x);
    }

    void _erase(int &x, int k) {
        if (key[x] == k) {
            if (cnt[x] > 1) {
                cnt[x]--;
            }   // 若节点不只1个副本,则副本数--
            else {
                if (childs[x][0] == 0 && childs[x][1] == 0) {
                    x = 0;
                    return;
                }   // 若没有儿子节点直接删除并返回
                // 否则需要将儿子旋转上来,再递归地进行删除
                int t = (priority[childs[x][0]] > priority[childs[x][1]]);
                rotate(x, t);
                _erase(x, k);
            }
        }
        else {  // 根据BST性质查找目标值
            _erase(childs[x][key[x] < k], k);
        }
        update(x);
    }

    int _getKth(int &x, int k) {
        // 利用子树大小信息查找第k大值
        if (k <= size[childs[x][0]]) {
            return _getKth(childs[x][0], k);
        }
        k -= size[childs[x][0]] + cnt[x];
        if (k <= 0) {
            return key[x];
        }
        return _getKth(childs[x][1], k);
    }

    int _getRank(int x, int k) {
        if (!x) return 0;
        if (k == key[x]) return size[childs[x][0]] + 1;
        else if (k < key[x]) return _getRank(childs[x][0], k);
        else return size[childs[x][0]] + cnt[x] + _getRank(childs[x][1], k);
    }

    int getPre(int k) {
        int x = root, pre;
        while (x)
        {
           if (key[x] < k) { pre = key[x], x = childs[x][1]; }
            else x = childs[x][0];
        }
        return pre;
    }
    int getNxt(int k) {
        int x = root, nxt;
        while (x)
        {
           if (key[x] > k) { nxt = key[x], x = childs[x][0]; }
            else x = childs[x][1];
        }
        return nxt;
    }

    void insert(int k) {
        _insert(root, k);
    }
    void erase(int k) {
        _erase(root, k);
    }
    int getKth(int k) {
        return _getKth(root, k);
    }
    int getRank(int k) {
        return _getRank(root, k);
    }
} mTreap;

// luogu P3369
void solve() {
    int n; cin >> n;
    int op, x;
    while (cin >> op >> x) {
        switch (op)
        {
        case 1:
            mTreap.insert(x);
            break;
        case 2:
            mTreap.erase(x);
            break;
        case 3:
            cout << mTreap.getRank(x) << '\n';
            break;
        case 4:
            cout << mTreap.getKth(x) << '\n';
            break;
        case 5:
            cout << mTreap.getPre(x) << '\n';
            break;
        case 6:
            cout << mTreap.getNxt(x) << '\n';
            break;
        }
    }
}

双端队列

这个问题我在数据结构与算法课上想过,并且在知乎上还提问了233。后来在校赛上遇到了同样的问题!当时手上一本上交的《ACM国际大学生程序设计竞赛 算法与实现》,看到了Treap可以查询第k大数的操作,当时灵机一动似乎可以搞,在赛场上解决了这个脑海里悬而未决的问题。实际上,所有平衡树通过维护子树大小,都可以实现查询第k大数的操作。不过,个人感觉Treap仅仅在BST的基础上增加priority一个随机属性并辅以旋转操作维护堆的性质,实在妙哉!

问题描述:维护一个双端队列,支持两个操作,1 x输出第x个元素,并将其移至队尾,2 x输出第x个元素,并将其移至队首。队内初始有n个元素,共有m次操作,n,m均是5e5的数量级。

我的做法:通过一个哈希表维护<key, element>,操作1通过getKth操作获取到该元素并输出,可以通过设置key为当前最小的key减1再将其插入Treap中,那么它将是Treap中最小的元素,也就是队首元素。操作2同理。

代码如下,Treap类省略。

unordered_map<int, int> mp;

void solve() {
    int n, m;
    cin >> n >> m;
    int a;
    for (int i = 1; i <= n; i++) {
        cin >> a;
        mp[i] = a;
        mTreap.insert(i);
    }
    int k, x;
    int l = 1, r = n;
    for (int i = 1; i <= m; i++) {
        cin >> k >> x;
        if (k == 1) {
            int ret = mTreap.getKth(x);
            cout << mp[ret] << '\n';
            mTreap.erase(ret);
            r++;
            int tmp = mp[ret];
            mp.erase(ret);
            mp[r] = tmp;
            mTreap.insert(r);
        }
        else {
            int ret = mTreap.getKth(x);
            cout << mp[ret] << '\n';
            mTreap.erase(ret);
            l--;
            int tmp = mp[ret];
            mp.erase(ret);
            mp[l] = tmp;
            mTreap.insert(l);
        }
    }
}

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

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

相关文章

29从零开始学Java之如何正确创建Java里的类?

作者&#xff1a;孙玉昌&#xff0c;昵称【一一哥】&#xff0c;另外【壹壹哥】也是我哦 千锋教育高级教研员、CSDN博客专家、万粉博主、阿里云专家博主、掘金优质作者 前言 在上一篇文章中&#xff0c;壹哥给大家介绍了面向对象和面向过程的概念&#xff0c;并介绍了两者的区…

【MySQL】数据库的基本操作三:增删改查进阶

目录 &#x1f31f;一、数据库约束 &#x1f308;1、Null约束&#xff1a;创建表时&#xff0c;可以指定某列不能为空。 &#x1f308;2、Unique约束&#xff1a;唯一约束 &#x1f308;3、Default约束&#xff1a;默认值约束 &#x1f308;4、Primary Key&#xff1a;主键…

@RefreshScope 动态刷新机制

前言 一般在项目中&#xff0c;我们集成Nacos做统一配置管理&#xff0c;同时配置支持动态刷新&#xff0c;项目中都会用到RefreshScope注解&#xff0c;这里和大家一起看看RefreshScope实现动态刷新的原理。 Scope注解 RefreshScope 能实现动态刷新全仰仗着Scope 这个注解&…

Matlab进阶绘图第19期—三角气泡热图

三角气泡热图&#xff0c;顾名思义&#xff0c;就是仅保留气泡热图数据矩阵的上三角或下三角部分。 三角气泡热图简单明了&#xff0c;通过不同颜色、不同大小的圆形表示数据的大小&#xff0c;可以更加直观地对矩阵数据进行可视化表达。 本文使用自制的tribubbleheatmap小工…

LSF/MM/BPF Summit 2023

5月8号&#xff0c;今年度的Linux Storage, Filesystem, Memory Management & BPF Summit已经拉开帷幕&#xff0c;Linux存储、文件系统、内存管理以及BPF领域的年度峰会又一次到来。此次峰会聚集了Linux最重要的开发专家以及内核子系统维护者&#xff0c;以规划和探索改进…

Arcgis Server/GeoServer服务启动后内存使用高,系统卡

说明 ArcGIS Server和geoserver本质上都是Tomcat服务&#xff0c;所以只需要设置Tomcat最大堆大小和最大内存大小就可以。此方法通用与Tomcat自身配置。 配置文件位置 一.Tomcat 安装目录/bin 二.ArcGIS Server .\ArcGIS\Server\framework\runtime\tomcat\bin 三.Geoserver …

【代码随想录】刷题Day22

1.二叉搜索树的公共祖先 235. 二叉搜索树的最近公共祖先 不同于普通二叉树&#xff0c;二叉搜索树得益于其顺序结构&#xff0c;其公共祖先的查找也有迹可循。自顶向下递归遍历&#xff0c;只要一个节点的val夹在p和q之间&#xff0c;那么该节点就是最近公共祖先。 1.首先公共…

HTML 中的常用标签用法

目录 1.基本结构 2.注释标签 3.标题标签 4.换行标签 5.格式化标签 6.图片标签 7.超链接标签 8.表格标签 合并单元格 9.列表标签 10.表单标签 form标签 input标签 补充 无语义标签 总 HTML是一种超文本标记语言,在网站上看到的信息都是它实现的.(是由标签所构…

RK3588平台开发系列讲解(内存篇)Linux 伙伴系统数据结构

平台内核版本安卓版本RK3588Linux 5.10Android 12文章目录 一、 页二、区三、内存节点沉淀、分享、成长,让自己和他人都能有所收获!😄 📢Linux 系统中,用来管理物理内存页面的伙伴系统,以及负责分配比页更小的内存对象的 SLAB 分配器了。 本篇将介绍伙伴系统相关数据结…

M304A_S905L3-B_5621无线蓝牙_当贝纯净桌面-线刷固件包

M304A_S905L3-B_5621无线蓝牙_当贝纯净桌面-线刷固件包 特点&#xff1a; 1、适用于对应型号的电视盒子刷机&#xff1b; 2、开放原厂固件屏蔽的市场安装和u盘安装apk&#xff1b; 3、修改dns&#xff0c;三网通用&#xff1b; 4、大量精简内置的没用的软件&#xff0c;运…

springboot第19集:权限

article 文章表sys_permission 后台权限表sys_role 后台角色表sys_role_permission 角色-权限关联表sys_user 用户表sys_user_role 用户-角色关联表 image.png image.png sys_user_role id user_id(用户id) role_id(角色id) sys_role id role_name(角色名) create_time(创建时间…

决策树生成剪枝算法原理

决策树生成算法 首先明确信息熵 信息增益的概念 信息增益表示得知特征X信息是的类Y的信息不确定性减少的程度 H(D) 经验熵表示对数据D进行分类的不确定性 H(D|A)经验条件熵表示对特征A给定条件下对数据集D进行分类的不确定性&#xff08;显然这个值越小越好 那么g(D,A)信息…

基于粒子群优化的中文文本分类

基本思路&#xff1a; 方法&#xff1a;使用优化算法&#xff08;如粒子群&#xff09;优化支持向量机SVM&#xff1b; 本文所使用的应用背景&#xff1a;中文文本分类&#xff08;同时可以应用到其他背景领域&#xff0c;如&#xff09; 应用背景&#xff08;元启发式算法优…

(学习日记)2023.5.9

写在前面&#xff1a; 由于时间的不足与学习的碎片化&#xff0c;写博客变得有些奢侈。 但是对于记录学习&#xff08;忘了以后能快速复习&#xff09;的渴望一天天变得强烈。 既然如此 不如以天为单位&#xff0c;以时间为顺序&#xff0c;仅仅将博客当做一个知识学习的目录&a…

QTableview常用几种代理总结

QTableview常用几种代理总结 [1] QTableview常用几种代理总结1、QCheckBox和QRadioButton的嵌入2、QHeadView中嵌入QCheckBox类3、QCombobox的嵌入4、 QCombox QCheckBox类5、SpinBox的嵌入类6、QProcess的嵌入类7、QProcess绘制版本的嵌入类8、QPushButton/QLabel/QImage的嵌…

鸿蒙Hi3861学习八-Huawei LiteOS-M(事件标记)

一、简介 事件是一种实现任务间通信的机制&#xff0c;可用于实现任务间的同步。但事件通信只能是事件类型的通信&#xff0c;无数据传输。一个任务可以等待多个事件的发生&#xff1a;可以是任意一个事件发生时唤醒任务进行事件处理&#xff1b;也可以是几个事件都发生后才唤醒…

mongodb副本集搭建

1.本次搭建使用三台centos7主机搭建集群&#xff0c;关闭防火墙和selinux服务 2.主机信息如下图所示 主机名称IPPortServiceA10.1.60.11427017mongodbB10.1.60.11527017mongodbC10.1.60.11827017mongodb 3.从官网下载mongodb安装包(我这里下载的是6.0.5版本的tgz包) Instal…

小家电LED显示驱动多功能语音芯片IC方案 WT2003H4 B002

随着时代的进步&#xff0c;智能家电的普及已经成为了一个趋势。而在智能家电中&#xff0c;LED显示屏也成为了不可或缺的一部分。因此&#xff0c;在小家电的设计中&#xff0c;LED显示驱动芯片的应用也越来越广泛。比如&#xff1a;电饭煲、电磁炉、数字时钟、咖啡机、电磁炉…

【Vue3】如何创建Vue3项目及组合式API

文章目录 前言 一、如何创建vue3项目&#xff1f; ①使用 vue-cli 创建 ②使用可视化ui创建 ③npm init vite-app ④npm init vuelatest 二、 API 风格 2.1 选项式 API (Options API) 2.2 组合式 API (Composition API) 总结 前言 例如&#xff1a;随着前端领域的不断发展&am…

【SSM框架】SpringMVC 中常见的注解和用法

SSM框架 SpringMVC 中常见的注解和用法基础注解介绍RequestMapping 注解介绍PostMapping 和 GetMapping 注解介绍 获取参数相关注解的介绍只通过 RequestMapping 来获取参数只传递一个参数传递对象参数传递多个参数(非对象) RequestParam 后端参数重命名required 必传参数的设置…