当前位置: 首页 > article >正文

数据结构红黑树

红黑树是一种自平衡的二叉搜索树,它通过确保任何从根到叶子的路径上不会有两个连续的红节点并且从根到叶子的所有路径上有相同数量的黑节点,从而近似平衡。这种平衡保证了在最坏情况下插入、删除、查找操作都能在O(log n)时间复杂度内完成。

下面,我将逐步介绍红黑树的关键操作,包括节点的定义、插入操作以及调整(修复)操作。由于完整的源码和解析非常冗长,我将简要概述每个部分,并给出关键代码片段。

红黑树节点的定义

在实现红黑树之前,我们首先定义树中的节点,包括节点的颜色、键值、左右子节点以及父节点的引用:

class RedBlackNode<T extends Comparable<T>> {
    enum Color { RED, BLACK }

    T key;
    Color color;
    RedBlackNode<T> left;
    RedBlackNode<T> right;
    RedBlackNode<T> parent;

    public RedBlackNode(T key, Color nodeColor, RedBlackNode<T> left, RedBlackNode<T> right) {
        this.key = key;
        this.color = nodeColor;
        this.left = left;
        this.right = right;
        this.parent = null; // 父节点在插入时确定
    }
}

红黑树的插入操作

插入操作首先按照二叉搜索树的性质插入节点,之后会根据红黑树的性质进行一系列的调整。

public void insert(T key) {
    RedBlackNode<T> node = new RedBlackNode<T>(key, RedBlackNode.Color.RED, null, null);
    // 正常的BST插入操作
    if (root == null) {
        root = node;
    } else {
        insertNode(root, node);
    }
    // 插入后的调整操作
    fixAfterInsertion(node);
}

插入后的调整操作

插入节点后可能会违反红黑树的性质,因此需要进行调整。下面是调整操作的伪代码概述:

while (新节点不是根节点且新节点的父节点是红色) {
    if (父节点是祖父节点的左子节点) {
        uncle = 祖父节点的右子节点
        if (uncle是红色) {
            // Case 1: 叔叔节点也是红色
            父节点设为黑色
            叔叔节点设为黑色
            祖父节点设为红色
            新节点 = 祖父节点
        } else {
            if (新节点是父节点的右子节点) {
                // Case 2: 新节点是其父节点的右子节点
                新节点 = 父节点
                左旋转(新节点)
            }
            // Case 3: 新节点是其父节点的左子节点
            父节点设为黑色
            祖父节点设为红色
            右旋转(祖父节点)
        }
    } else {
        // 对称地处理父节点是祖父节点右子节点的情况
        ...
    }
}
根节点设为黑色

左旋转示例

左旋转是在插入节点导致树不平衡时重新平衡树的操作之一。以下是左旋转的概念性Java代码:

void leftRotate(RedBlackNode<T> x) {
    RedBlackNode<T> 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;
    } else if (x == x.parent.left) {
        x.parent.left = y;
    } else {
        x.parent.right = y;
    }
    y.left = x;
    x.parent = y;
}

右旋转示例

右旋转是左旋转的对称操作,用于重新平衡树。

void rightRotate(RedBlackNode<T> y) {
    RedBlackNode<T> 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.right) {
        y.parent.right = x;
    } else {
        y.parent.left = x;
    }
    x.right = y;
    y.parent = x;
}

细节和注意事项

  1. 颜色翻转: 在某些情况下,你需要改变父节点和叔叔节点的颜色,并进行旋转。
  2. 插入时的颜色: 新插入的节点总是红色的。
  3. 根节点的颜色: 每次插入或删除后,根节点应保持为黑色。
  4. 旋转:旋转操作是用来重新平衡树结构的。左旋是将节点向左下方转移,而右旋是将其向右下方转移。

实现红黑树的完整代码需要大量的时间和精力。通常情况下,大多数编程语言的标准库中已经包含了红黑树的实现,比如Java中的TreeMapTreeSet。如果你要实现自己的红黑树,建议详细阅读相关书籍或文档,并在严格控制下一步步实现。


http://www.kler.cn/a/233861.html

相关文章:

  • 用uniapp写一个播放视频首页页面代码
  • 黑马JavaWeb开发跟学(十四).SpringBootWeb原理
  • 物联网控制期末复习
  • ==和===的区别,被坑的一天
  • shell学习数学运算符和字符串(三)
  • Mac 安装Mysql启动Mysql以及数据库的常规操作
  • 分布式搜索引擎 elasticsearch
  • MySQL-视图(VIEW)
  • 基于Chrome插件的Chatgpt对话无损导出markdown格式(Typora完美显示)
  • 【数据结构】双向链表(链表实现+测试+原码)
  • 练习:鼠标类设计之1_类内容解析
  • JVM-双亲委派机制
  • 一步步建立一个C#项目(连续读取S7-1200PLC数据)
  • 监控概述、安装zabbix、配置zabbixagent、添加被控端主机、常用监控指标、自定义监控项
  • 2024年华为OD机试真题-密码解密-Java-OD统一考试(C卷)
  • 精灵图,字体图标,CSS3三角
  • css2复合选择器
  • 云计算市场分析
  • 深入探索MySQL的虚拟列:发展、原理与应用
  • 吉他学习:右手拨弦方法,右手拨弦训练 左手按弦方法
  • VSCode:替换空行
  • 【计算机网络】协议层次及其服务模型
  • 结构体的大小以及内存对齐问题
  • 搭建yum仓库服务器
  • C++ dfs 的状态表示(五十一)【第十一篇】
  • vue学习——集成sass