数据结构之平衡树:红黑树的介绍与Python代码实现——17

红黑树的介绍与Python代码实现

红黑树的介绍

  • 红黑树(Red-Black Tree)是一种平衡二叉查找树,它是一种以比较简单的方式实现的2-3查找树

红黑树基于2-3查找树的表现

  • 红链接:将两个2-结点连接起来构成一个3-结点 ;
  • 黑链接:则是2-3树中的普通链接。
    在这里插入图片描述

红黑树的定义:
红黑树是含有红黑链接并满足下列条件的二叉查找树: .

  1. 红链接均为左链接;
  2. 没有任何一个结点同时和两条红链接相连;
  3. 该树是完美色平衡的,即任意空链接到根结点的路径上的黑链接数量相同;

红黑树的优点:

  • 一颗二叉树,每一个结点只需要额外多一位空间即可实现红黑树,这一位空间通常用于存放和表示红黑结点,而这些红黑标识则可以用来使红黑树保持接近平衡的状态
  • 记录每一个结点的红黑状态,只需要额外的一位空间,这使得红黑树的储存空间大小在一定程度上可以认为和无颜色标记的二叉树的储存空间大小等同,在大多数情况下,无需额外的储存成本就能储存着一位的红黑记录信息
  • 红黑树不是完美的平衡二叉树,但是它的平衡状态足够让我们能很方便地进行搜寻操作,红黑树的查询、插入、删除操作时间复杂度都是O(log n)

红黑树的平衡化

为什么需要平衡化?

  • 在对红黑树进行一些增删改查的操作后 ,很有可能会出现红色的右链接或者两条连续红色的链接,而这些都不满足红黑树的定义,所以我们需要对这些情况通过旋转进行修复,让红黑树保持平衡。

平衡化的方法

  1. 左旋
  2. 右旋
左旋

时机:

  • 当某个结点的左子结点为黑色,右子结点为红色,此时需要左旋。

实现方式:

  • 当前节点为h,它的右节点是x;
    color的值是由父结点指过来的线的颜色
    在这里插入图片描述

实现过程:

在这里插入图片描述

右旋

时机:

  • 当某个结点的左子结点是红色,并且左子结点的左子结点也是红色,要右旋

实现方式

  • 前提:当前结点为h ,它的左子结点为x ;
    color的值由父结点指过来的线的颜色
    在这里插入图片描述
    实现过程:
    在这里插入图片描述
    右旋之后保持了有序性;但是红链接连接了三个结点不满足2-3树的性质,同时违背了红黑树右链接不能为红链接的要求,这个问题下面将会介绍使用颜色反转的方法来解决

平衡步骤

  1. 向单个2-结点中插入新键后,结果是插入到该结点的右子结点,则需要进行左旋:
    在这里插入图片描述
    将c结点替换b(使b=c),b的颜色变为红,然后让b称为c的左子节点即可完成左旋(根结点的颜色后面会有一个操作让其始终保持黑色)
  2. 向底部的2-结点插入新键
    在这里插入图片描述
    情况同一,只是需要多一步,左旋之后,C的颜色要变为黑色(表示指向C的边为红色)
  3. 颜色反转
    在这里插入图片描述
    当一个结点的左子结点和右子结点的color都为RED时, 也就是出现了临时的4-结点,此时只需要把左子结点和右子结点的颜色变为BLACK ,同时让当前结点的颜色变为RED即可。
  4. 向一棵双键树(即一个3-结点)中插入新键
    可分为三种子情况
    4-1. 新键大于原树中的两个键:
    在这里插入图片描述
    4-2. 新键小于原树中的两个键
    在这里插入图片描述
    4-3. 新键介于原数中两个键之间
    在这里插入图片描述
  5. 根结点的颜色总是黑色
    在每次放入元素的操作完成之后,将根结点的颜色变更为’Black’即可:
    self.root.color = 'Black'
  6. 向树底部的3-结点插入新键
    在这里插入图片描述

操作方法

  1. is_red(node) 判断传入的结点node是否为红色
  2. rotate_left(node) 将传入的结点进行左旋操作
  3. rotate_right(node) 将传入的结点进行右旋操作
  4. alter_color(node) 将传入的结点进行颜色反转操作
  5. put(key, val) 插入一个键为key,值为val的元素,插入之后自动按键进行排序
  6. get_value(key) 根据传入的键key,获取对应结点的值

Python代码实现

二叉树结点设计

class Node:def __init__(self, key, value):self.key = keyself.value = valueself.left = Noneself.right = Noneself.color = False

功能实现

class RedBlackTree:def __init__(self):self.root = Noneself.N = 0def size(self):return self.Ndef is_red(self, node):return str(node.color).lower() == 'red' if node else Falsedef rotate_left(self, node):"""Rotate left when the edge from the current node to its right child node is red"""# h is the current nodeh = node# x is the current node's right childx = node.righth.right = x.leftx.left = hx.color = h.colorh.color = 'Red'return xdef rotate_right(self, node):"""Rotate right when both the left edge and the left child's left edge are red"""h = nodex = node.lefth.left = x.rightx.right = hx.color = h.colorh.color = 'Red'return xdef alter_color(self, node):"""Alter a node's color"""node.color = 'Red'node.left.color = 'Black'node.right.color = 'Black'def put(self, key, val):"""Put an element into this tree"""def put_into(node, key, val):if not node:return Node(key, val)# Rank the orderif key < node.key:  # Recursively to compare key with its left childnode.left = put_into(node.left, key, val)elif key > node.key:node.right = put_into(node.right, key, val)else:   # Swap their the node.value with valnode.value = valreturn node# Rotation or alter colorif self.is_red(node.right) and not self.is_red(node.left):# Rotate leftself.rotate_left(node)if self.is_red(node.left) and self.is_red(node.left.left):# Rotate rightself.rotate_right(node)if self.is_red(node.left) and self.is_red(node.right):# Alter colorself.alter_color(node)self.N += 1return nodeself.root = put_into(self.root, key, val)self.root.color = 'Black'return self.rootdef get(self, key):"""Get a value according to the given key"""def get_value(node, key):if not node:returnif key < node.key:return get_value(node.left, key)elif key > node.key:return get_value(node.right, key)else:return node.valueval = get_value(self.root, key)return val

代码测试

if __name__ == '__main__':RBT = RedBlackTree()RBT.put(1, 'G')RBT.put(2, 'K')RBT.put(3, 'd')RBT.put(3, 'D')for i in range(1, 4):print(RBT.get(i), end=' ')print('\n', RBT.size())print(RBT.root.color)print(RBT.root.key, RBT.root.value)print(RBT.root.right.key, RBT.root.right.value)print(RBT.root.right.right.key, RBT.root.right.right.value)

测试结果

G K D 5
Black
1 G
2 K
3 D

插入的元素都按照对应的键获取到了,说明代码没有什么问题

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

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

相关文章

Python键鼠自动化函数

自动化之图形界面库pyautogui_jgdabc的博客-CSDN博客 https://blog.csdn.net/jgdabc/article/details/121952632

转:绝对干货--WordPress自定义查询wp_query所有参数详细注释

1 <?php2 /**3 * WordPress 查询综合参考4 * 编译&#xff1a;luetkemj - luetkemj.com5 *6 * 官方文档: http://codex.wordpress.org/Class_Reference/WP_Query7 * 源代码: http://core.trac.wordpress.org/browser/tags/3.5.1/wp-includes/query.php8 */9 10 $args arra…

数据结构之并查集:并查集的介绍与Python代码实现——18

并查集的介绍 并查集&#xff08;Union-find&#xff09;数据结构也称作合并查找集&#xff08;Merge-find set&#xff09;或者不相交集数据结构&#xff08;disjoint-set data structure&#xff09;&#xff0c;它是一种记录了由一个或多个元素组成的不连续的分组的集合。并…

我的合伙人

我突然想写一篇为《我的合伙人》的文章。 为什么要写这篇文章&#xff0c;因为我发现在我们的生活中&#xff0c;能找到一个人做为自己的合伙人是很困难的。 我在看《富爸爸&#xff0c;穷爸爸》这本书的时候发现&#xff0c;基米和他的朋友组成了合伙人&#xff0c;他们有共同…

图像特征提取与描述_角点特征01:Harris算法+Shi-Tomas算法

1 Harris角点检测 1.1 原理 Harris角点检测的思想是通过图像的局部的小窗口观察图像&#xff0c;角点的特征是窗口沿任意方向移动都会导致图像灰度的明显变化&#xff0c;如下图所示&#xff1a; 将上述思想转换为数学形式&#xff0c;即将局部窗口向各个方向移动(u,v)并计算…

canvas小程序-快跑程序员

canvas不用说html5带来的好东西&#xff0c;游戏什么的&#xff0c;么么哒 记得有一天玩手机游戏&#xff0c;就是一个跳跃过柱子那种&#xff0c;其实元素很简单啊&#xff0c;app能开发&#xff0c;借助html5 canvas也可以啊&#xff0c;于是就开始了。 --------------------…

数据结构之并查集:UF-Tree优化并查集——19

并查集的优化 在上一节了解到并查集的快速查询&#xff0c;合并&#xff0c;判断归属组等操作&#xff0c;虽然这些操作都非常方便&#xff0c;但是在数据量较大的情况下&#xff0c;并查集的效率并不算高&#xff1a; 上一节中实现代码中使用的合并方法(merge&#xff0c;AP…

图像特征提取与描述_角点特征02:SIFT算法+SURF算法

SIFT/SURF算法 1.1 SIFT原理 前面两节我们介绍了Harris和Shi-Tomasi角点检测算法&#xff0c;这两种算法具有旋转不变性&#xff0c;但不具有尺度不变性&#xff0c;以下图为例&#xff0c;在左侧小图中可以检测到角点&#xff0c;但是图像被放大后&#xff0c;在使用同样的窗…

linux内核定时器使用

//加入头文件 #include <linux/hrtimer.h> //init owen.wei modify time struct hrtimer timer_ptt; //初始化定时器 hrtimer_init(&timer_ptt, CLOCK_MONOTONIC, HRTIMER_MODE_REL); time

图像特征提取与描述_角点特征03:Fast算法+ORB算法

1 Fast算法 1.1 原理 我们前面已经介绍过几个特征检测器&#xff0c;它们的效果都很好&#xff0c;特别是SIFT和SURF算法&#xff0c;但是从实时处理的角度来看&#xff0c;效率还是太低了。为了解决这个问题&#xff0c;Edward Rosten和Tom Drummond在2006年提出了FAST算法&…

windows xp 下的putty不能使用小键盘的问题

change setting->Features->Disable application keypad mode (勾选上这个)

数据结构之并查集:路径压缩继续优化并查集——20

路径压缩继续优化并查集 在实现的并查集中&#xff0c;在合并操作merge(item1, item2)时&#xff0c;会不管两个元素所在的分组大小&#xff0c;总是将item 1的分组合并到item2的分组&#xff0c;这样可能会导致树的深度无必要地增加&#xff1a; 如果是大树合并到小树上&…

Pycharm使用matplotlib绘图时无法显示中文问题

在画图语句前&#xff0c;加上以下两行代码&#xff1a; plt.rcParams[font.sans-serif] [uSimHei] plt.rcParams[axes.unicode_minus] False

caffe各种依赖包配置

从15号配置完了云服务器的FTP服务后就一直在弄caffe。 原以为会很简单的yum install 几个依赖包。然后下载caffe并且make 编译一下就可以了。 结果万万没想到我还是too young too naive! 依赖包大部分只能靠手装,因为yum install 有很大几率装不上这些依赖包&#xff01; 以下是…

高通8X25Q wifi BT 调试文档

1、蓝牙调试 主要是蓝牙的I2C地址跟 地磁仪的I2C地址冲突&#xff0c;被地磁仪给占用了&#xff0c;改回来就好了。 --- a/android-lte/kernel/arch/arm/mach-msm/msm8x25/goso-msm7627a-bt.c b/android-lte/kernel/arch/arm/mach-msm/msm8x25/goso-msm7627a-bt.c -981,8 98…

数据结构之并查集:并查集解决案例, Python——21

并查集解决案例畅通工程 案例问题介绍&#xff1a; 某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇。省政府"畅通工程”的目标是使全省任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要互相间接通过道路可达即可)。问最少还…

图像特征提取与描述_角点特征04:LBP算法+HOG特征算子

1.LBP算法 LBP(Local Binary Pattern)指局部二值模式&#xff0c;是一种用来描述图像局部特征的算子&#xff0c;LBP特征具有灰度不变性和旋转不变性等显著优点。它是由T. Ojala, M.Pietikinen, 和 D. Harwood在1994年提出&#xff0c;由于LBP特征计算简单、效果较好&#xff…

判断两个字符串是否由相同的字符组成

描述 判断两个字符串是否由相同的字符组成 分析 方法一&#xff0c;排序法。对两个字符串进行排序&#xff0c;然后在比较。 方法二&#xff0c;空间换时间。ascII字符共256个&#xff0c;对字符串1出现的字符在对应的数组里加1&#xff0c;对字符串1出现的字符在对应的数组里减…

Android input keyevent

adb shell input keyevnet 3 点击返回键操作 adb shell input keyevent 4 点击home键操作 ./frameworks/base/core/java/android/view/KeyEvent.java 这里面有的keyevent都可以用得上。

视频操作_01视频读写:视频读写+读取视频+保存视频

1 从文件中读取视频并播放 在OpenCV中我们要获取一个视频&#xff0c;需要创建一个VideoCapture对象&#xff0c;指定你要读取的视频文件&#xff1a; 1.创建读取视频的对象 cap cv.VideoCapture(filepath) 参数&#xff1a; filepath: 视频文件路径 2.视频的属性信息 2.1…