【C语言】红黑树详解以及C语言模拟

    • 一、红黑树的性质
    • 二、红黑树的旋转操作
    • 三、红黑树的插入操作
    • 四、红黑树的删除操作
    • 五、红黑树的应用
    • 六、C语言模拟红黑树
    • 七、总结

 红黑树是一种自平衡二叉查找树,它能够保持树的平衡,从而确保查找、插入和删除的最坏情况时间复杂度为O( l o g n log_n logn)。在红黑树中,节点被涂成红色或黑色,并且通过旋转和重新着色等操作来保持树的平衡。本文将详细介绍红黑树的原理,并使用C语言进行模拟实现。

一、红黑树的性质

请添加图片描述

红黑树具有以下性质:

  1. 每个节点非红即黑。
  2. 根节点是黑色的。
  3. 每个叶子节点(NIL)是黑色的。
  4. 如果一个节点是红色的,则它的子节点必须是黑色的(不能有两个连续的红色节点)。
  5. 从任一节点到其每个叶子节点的所有路径都包含相同数量的黑色节点。

二、红黑树的旋转操作

红黑树通过旋转操作来维护树的平衡。旋转分为左旋和右旋两种操作。

  1. 左旋:将某个节点作为旋转中心,将其右子节点向上移动,代替该节点的位置,然后将该节点作为左子节点。
  2. 右旋:将某个节点作为旋转中心,将其左子节点向上移动,代替该节点的位置,然后将该节点作为右子节点。

三、红黑树的插入操作

红黑树的插入操作包括以下几个步骤:

  1. 执行二叉查找树的插入操作,将新节点涂成红色。
  2. 调整红黑树,使其满足红黑树的性质。
    插入操作可能导致以下几种情况:
  3. 叔叔节点是红色:此时,只需重新着色即可。
  4. 叔叔节点是黑色或不存在:
    (1)当前节点是父节点的左子节点,且父节点是祖父节点的左子节点,或者当前节点是父节点的右子节点,且父节点是祖父节点的右子节点。此时,只需进行一次旋转操作。
    (2)当前节点是父节点的左子节点,且父节点是祖父节点的右子节点,或者当前节点是父节点的右子节点,且父节点是祖父节点的左子节点。此时,需要进行两次旋转操作。

四、红黑树的删除操作

红黑树的删除操作包括以下几个步骤:

  1. 执行二叉查找树的删除操作。
  2. 调整红黑树,使其满足红黑树的性质。
    删除操作可能导致以下几种情况:
  3. 被删除节点是红色:直接删除,不影响红黑树的性质。
  4. 被删除节点是黑色:
    (1)被删除节点的子节点是红色:直接删除,并将子节点涂成黑色。
    (2)被删除节点的子节点是黑色:需要进行调整,以保持红黑树的性质。

五、红黑树的应用

 红黑树在计算机科学中有着广泛的应用,例如在Java的TreeMap和TreeSet中,以及C++的STL中。红黑树能够提供高效的查找、插入和删除操作,因此在需要维护有序数据集合的场景中,红黑树是一个很好的选择。

六、C语言模拟红黑树

以下是一个简单的C语言实现红黑树的示例:

#include <stdio.h>
#include <stdlib.h>typedef enum {RED, BLACK} Color;typedef struct Node {int data;Color color;struct Node *parent;struct Node *left;struct Node *right;
} Node;/*** 创建并初始化一个节点。* @param data 节点中要存储的数据。* @return 返回指向新创建节点的指针。*/
Node *createNode(int data) {// 分配内存给新节点Node *newNode = (Node *)malloc(sizeof(Node));if (newNode == NULL) {printf("Memory allocation failed!\n"); // 内存分配失败处理exit(1);}newNode->data = data; // 设置节点数据newNode->color = RED; // 新插入的节点默认为红色newNode->parent = NULL; // 初始化父节点为NULLnewNode->left = NULL; // 初始化左子节点为NULLnewNode->right = NULL; // 初始化右子节点为NULLreturn newNode; // 返回新创建的节点
}/*** 左旋操作,用于平衡二叉搜索树(BST)。* * @param root 指向当前树根节点的指针的地址。* @param x 需要进行左旋操作的节点。* 说明:函数不返回任何值,但会修改树的结构。*/
void leftRotate(Node **root, Node *x) {Node *y = x->right;x->right = y->left;if (y->left != NULL)y->left->parent = x;y->parent = x->parent;if (x->parent == NULL)*root = y; // 如果x是根节点,更新根节点。else if (x == x->parent->left)x->parent->left = y; // 如果x是其父节点的左子节点,更新左子节点。elsex->parent->right = y; // 如果x是其父节点的右子节点,更新右子节点。y->left = x;x->parent = y;
}/*** 右旋操作* * @param root 指向当前树根节点的指针的地址。* @param x 需要进行右旋操作的节点。* 说明:函数不返回任何值,但会修改树的结构。*/
void rightRotate(Node **root, Node *y) {Node *x = y->left;y->left = x->right;if (x->right != NULL)x->right->parent = y;x->parent = y->parent;if (y->parent == NULL)*root = x;else if (y == y->parent->left)y->parent->left = x;elsey->parent->right = x;x->right = y;y->parent = x;
}/*** 插入修复函数* 该函数用于在红黑树中插入节点后,维护红黑树的性质。主要通过旋转和颜色的变换来保持红黑树的性质。* * @param root 指向红黑树根节点的指针的地址* @param z 待插入的节点*/
void insertFixup(Node **root, Node *z) {while (z != *root && z->parent->color == RED) {if (z->parent == z->parent->parent->left) {Node *y = z->parent->parent->right;if (y != NULL && y->color == RED) {z->parent->color = BLACK;y->color = BLACK;z->parent->parent->color = RED;z = z->parent->parent;} else {if (z == z->parent->right) {z = z->parent;leftRotate(root, z);}z->parent->color = BLACK;z->parent->parent->color = RED;rightRotate(root, z->parent->parent);}} else {Node *y = z->parent->parent->left;if (y != NULL && y->color == RED) {z->parent->color = BLACK;y->color = BLACK;z->parent->parent->color = RED;z = z->parent->parent;} else {if (z == z->parent->left) {z = z->parent;rightRotate(root, z);}z->parent->color = BLACK;z->parent->parent->color = RED;leftRotate(root, z->parent->parent);}}}(*root)->color = BLACK;
}/*** 向红黑树中插入一个新节点* * @param root 指向红黑树根节点的指针的地址* @param data 需要插入的数据*/
void insert(Node **root, int data) {Node *z = createNode(data);Node *y = NULL;Node *x = *root;while (x != NULL) {y = x;if (z->data < x->data)x = x->left;elsex = x->right;}z->parent = y;if (y == NULL)*root = z;else if (z->data < y->data)y->left = z;elsey->right = z;insertFixup(root, z);
}void inorderTraversal(Node *root) {if (root != NULL) {inorderTraversal(root->left);printf("%d ", root->data);inorderTraversal(root->right);}
}int main() {Node *root = NULL;insert(&root, 10);insert(&root, 20);insert(&root, 30);insert(&root, 15);insert(&root, 18);inorderTraversal(root);printf("\n");return 0;
}

完整代码(包括红黑树节点输出以及删除节点操作)【链接】

结果
在这里插入图片描述

七、总结

 红黑树是一种自平衡二叉查找树,它能够保持树的平衡,从而保证查找、插入和删除的最坏情况时间复杂度为O( l o g n log_n logn)。红黑树通过旋转操作和颜色变更来维护树的性质,适用于需要维护有序数据集合的场景。在Java和C++的STL中,红黑树得到了广泛的应用。

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

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

相关文章

LoRA: 大模型的低秩适配

笔记整理&#xff1a;陈一林&#xff0c;东南大学硕士&#xff0c;研究方向为不确定知识图谱规则学习 链接&#xff1a;https://arxiv.org/abs/2106.09685 1、动机 自然语言处理的一个重要范式包括在通用领域数据上进行大规模预训练&#xff0c;然后对特定任务或领域进行适应性…

JAVA10迭代更新

文章目录 1 概述2 语法层次的变化1_局部变量的类型推断 3 API层次的变化1_集合中新增copyOf创建只读集合2_Optional 新增了orElseThrow方法 4 其他1_垃圾回收器接口2_G1 并行 Full GC3_应用程序类数据共享(扩展 CDS 功能)4_实验性的基于 Java 的 JIT 编译器 1 概述 2018年3月21…

新能源汽车小米su7

小米su7汽车 function init() {const container document.querySelector( #container );camera new THREE.PerspectiveCamera( 20, window.innerWidth / window.innerHeight, 1, 50000 );camera.position.set( 0, 700, 7000 );scene new THREE.Scene();scene.background ne…

vue项目中基于fabric 插件实现涂鸦画布功能

vue项目中基于fabric 插件实现涂鸦画布功能 一、效果图二、安装依赖三、main.js引入四、主要代码 一、效果图 二、安装依赖 npm install fabric 三、main.js引入 import fabric from fabric Vue.use(fabric);四、主要代码 //封装成了一个组件 <template><el-dialogt…

FlashSpeech、ID-Animator、TalkingGaussian、FlowMap、CutDiffusion

本文首发于公众号&#xff1a;机器感知 FlashSpeech、ID-Animator、TalkingGaussian、FlowMap、CutDiffusion Gradient Guidance for Diffusion Models: An Optimization Perspective Diffusion models have demonstrated empirical successes in various applications and ca…

《MATLAB科研绘图与学术图表绘制从入门到精通》示例:绘制婴儿性别比例饼图

在MATLAB 中可以使用 pie 函数来创建饼图。饼图是一种展示不同部分占总体的相对比例的图表。 本示例从“婴儿出生数据.csv”文件读取婴儿出生数据&#xff0c;然后计算男性和女性婴儿的数量&#xff0c;使用MATLAB绘制饼图。 配套图书链接&#xff1a;https://item.jd.com…

AI图书推荐:AI驱动的图书写作工作流—从想法构思到变现

《AI驱动的图书写作工作流—从想法到变现》&#xff08;AI-Driven Book Creation: From Concept to Cash&#xff09;是Martynas Zaloga倾力打造的一本实用指南&#xff0c;它巧妙地将写作艺术与人工智能前沿技术相结合。此书不仅揭示了AI在图书出版领域的无限潜力&#xff0c;…

应用层协议 -- HTTPS 协议

目录 一、了解 HTTPS 协议 1、升级版的 HTTP 协议 2、理解“加密” 二、对称加密 1、理解对称加密 2、对称加密存在的问题 三、非对称加密 1、理解非对称加密 2、中间人攻击 3、CA 证书和数字签名 四、总结 一、了解 HTTPS 协议 1、升级版的 HTTP 协议 HTTPS 也是…

fatal: unable to access ‘https://github.com/alibaba/flutter_boost.git/

Git error. Command: git fetch stdout: stderr: fatal: unable to access ‘https://github.com/alibaba/flutter_boost.git/’: Failed to connect to github.com port 443 after 75005 ms: Couldn’t connect to server exit code: 128 GitHub (国际型)代码 分发平台/托管平…

Mycat(一)入门概述

文章目录 概述作用原理 Mycat1.x 与 Mycat2 功能对比1.x 与 2.0 功能对比图 Mycat2 相关概念概念描述 配置文件1、服务&#xff08;server&#xff09;2、用户&#xff08;user&#xff09;3、数据源&#xff08;datasource&#xff09;4、集群&#xff08;cluster&#xff09;…

车企的数智化“内功”,大模型帮修炼

文&#xff5c;白 鸽 编&#xff5c;王一粟 时隔4年回归的北京车展&#xff0c;遇上了中国智能汽车的热潮。 开年价格战的持续洗礼&#xff0c;不仅让一众中国车企都慌得一批&#xff0c;也让全球巨头特斯拉也面临一季度销量大跌局面。 与此同时&#xff0c;智能汽车还在…

C++初识内存管理和模版

目录 前言 1.C/C内存分布 2. C的内存管理方式 2.1 new/delete操作内置类型 2. new和delete操作自定义类型 3. operator new和operator delete函数 4. new和delete的实现原理 4.1 内置类型 4.2 自定义类型 5. malloc/free和new/delete的区别 6. 初识模版 6.1 泛型编…

ERROR: Permission denied (publickey,gssapi-keyex,gssapi-with-mic,password).

今天本来想在A服务器上传文件给B服务器的结果发现明明给root用户设置了密码就是远程登陆不了&#xff0c;后来才发现在容器中很多服务都是没有的&#xff0c;所以刚安装后忘记了修改配置文件&#xff0c;导致远程登陆失败。 报错&#xff1a; 解决方法&#xff1a; 在/etc/ssh…

申请高德地图,报错INVALID_USER_SCODE处理

配置上key后 报错&#xff1a; 解决&#xff1a;将应用类型修改为出行&#xff0c;问题解决 扩展&#xff1a;应用申请 进入应用管理&#xff0c;创建新应用&#xff08;这里我选了导航&#xff0c;就报了上边的错误&#xff09; 新应用中添加 key&#xff0c;服务平台选择…

static和extern关键字详解

目录 创作不易&#xff0c;如对您有帮助&#xff0c;还望一键三连&#xff0c;谢谢&#xff01;&#xff01;&#xff01; 回顾 1.作用域和声明周期 1.1作用域 1.2生命周期 2.static和extern 2.1extern 2.2static 2.2-1static修饰局部变量 2.2-2static修饰全局变量 创…

解决在服务器中减少删除大文件夹耗时太久的问题

在数据驱动的现代商业环境中&#xff0c;企业对服务器的高效运作有着极高的依赖性。然而&#xff0c;IT管理员们常常面临一个棘手的问题&#xff1a;删除服务器上的大型文件夹过程缓慢&#xff0c;这不仅降低了工作效率&#xff0c;还可能对用户体验造成负面影响。本文将介绍一…

2024 年 Rust 开发者路线图

Rust 近年来因其对性能、安全性和并发性的关注而广受欢迎。作为一名开发人员&#xff0c;掌握 Rust 可以为各种机会打开大门&#xff0c;包括 Web 开发。 在 github 上发现了这个优秀的路线图&#xff0c;由 Anshul Goyal 创建&#xff0c;它提供了一条全面的路径&#xff0c;概…

MIEC CS172(Prolog)

Chapter 1 and 2 Fact Facts: Facts are statements that areassumed to be true. The dot ‘.’ character must come at the end of a fact. Example: We want to tell “John likes Mary” : English interpretation The standard form of fact in Prolog Likes (john, ma…

怎么用AI绘画进行人物修复?

用过AI绘画生成人物图片的朋友们是不是都碰到过这样的问题&#xff1a;诡异的造型、崩坏的五官、离谱的手指头、乱七八糟的背景...指望AI一次性生成百分百完美的图貌似有点难啊。 现在AI绘画有了【脸部修复】【手部修复】功能&#xff0c;就能够轻松解决这些的问题了&#xff0…

Facebook的时间机器:回溯社交媒体的历史

1. 社交媒体的起源与早期模式 社交媒体的历史可以追溯到互联网的早期发展阶段。在Web 1.0时代&#xff0c;互联网主要是一个信息发布平台&#xff0c;用户主要是被动地接收信息。但随着Web 2.0的兴起&#xff0c;互联网逐渐转变为一个互动和参与的平台&#xff0c;社交媒体应运…