【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,一经查实,立即删除!

相关文章

批量控制教程-Ansible管理windows

背景 你厌恶要手动操作多台机器进行某些重复的操作吗&#xff1f;想象一下&#xff0c;在周五的晚上你想要下班了&#xff0c;但是你得在很多台机器手动发布一些东西&#xff0c;每台机器都要整半小时&#xff0c;整整8台机器&#xff0c;一晚上几个小时可以预见又没了。 ans…

LoRA: 大模型的低秩适配

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

CentOS命令大全

系统信息查询命令 在CentOS系统中&#xff0c;了解如何查询系统信息对于系统管理和性能监控至关重要。以下是一些基本而强大的命令&#xff0c;用于获取关于您的系统的各种信息。 1. uname - 获取系统信息 uname命令用于打印系统信息&#xff0c;例如内核版本、主机名等。使…

python之schedule

在Python中&#xff0c;可以使用schedule模块来执行定时任务。这个模块提供了简单易用的API&#xff0c;可以让你按照指定的时间间隔或特定时间点执行函数或任务。 首先&#xff0c;需要使用pip安装schedule模块&#xff1a; pip install schedule下面是一个简单的例子&#…

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…

Java中static关键字的使用与练习

Java中static关键字的使用与练习 在Java编程中&#xff0c;static关键字是一个非常重要的概念&#xff0c;它用于修饰类的成员变量和方法。通过static关键字&#xff0c;我们可以创建与类本身关联而不是与类的实例关联的成员。下面&#xff0c;我们将详细探讨static关键字在Ja…

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…

C语言中的指针常量的常量指针

指针常量和常量指针是C/C编程语言中两个重要的概念&#xff0c;它们都与指针有关&#xff0c;但具有不同的含义和用途。 1. 指针常量&#xff08;Pointer to Constant&#xff09; 指针常量指的是一个指针的值&#xff08;即它所指向的地址&#xff09;在初始化之后不能再被改…

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

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

Unity 帧同步游戏解决方案梳理

帧同步游戏解决方案梳理 一、保证所有客户端的计算结果一致二、帧同步手感优化&#xff1a;三、不同步问题总结&#xff1a;四、帧同步优化&#xff1a; 一、保证所有客户端的计算结果一致 保证所有客户端的计算结果一致 1、逻辑与显示分离 逻辑控制显示&#xff0c;而显示的执…

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 也是…

光端机——光纤通信学习笔记(八)

​​​​​​ 光发射机 基本性能要求 合适的发光波长 良好的消光比 合适的输出光功率 调制特性好 基本组成 光源 光源:是实现电光转换的关键器件&#xff0c;在很大程度上决定着光发射机的性能。 (1)发射的光波长应和光纤低损耗“窗口”一致&#xff0c;即中心波长应在 850…

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;…

XiaodiSec day035 Learn Note 小迪安全学习笔记

XiaodiSec day035 Learn Note 小迪安全学习笔记 记录得比较凌乱&#xff0c;不尽详细 day35 前言 关于 xss 入门 xss 开始 是一个前端的漏洞 数据输入输出的地方容易产生 分类: 反射性存储型DOM 型很多很多 产生过程使用<script>alert(1)</script>弹个窗 …

设计模式- 桥接模式(Bridge Pattern)结构|原理|优缺点|场景|示例

设计模式&#xff08;分类&#xff09; 设计模式&#xff08;六大原则&#xff09; 创建型&#xff08;5种&#xff09; 工厂方法 抽象工厂模式 单例模式 建造者模式 原型模式 结构型&#xff08;7种&#xff09; 适配器…

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

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