数据结构之平衡树:红黑树的介绍与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代码实现——18

并查集的介绍 并查集&#xff08;Union-find&#xff09;数据结构也称作合并查找集&#xff08;Merge-find set&#xff09;或者不相交集数据结构&#xff08;disjoint-set data structure&#xff09;&#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;在使用同样的窗…

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

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

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

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

数据结构之并查集:并查集解决案例, 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…

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

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

数据结构之图:无向图的介绍与功能实现,Python——22

无向图&#xff08;Undigraph&#xff09;的介绍 引入 生活中的图&#xff0c;有地图&#xff0c;集成电路板的图&#xff0c;可以看类似的看做是数据结构中的图数据有"一对一"&#xff0c;“一对多”和“多对多”的关系&#xff0c;前两种分别表示线性表和树的存储…

视频操作_02视频追踪:meanshift算法+Camshift算法

1.meanshift 1.1原理 meanshift算法的原理很简单。假设你有一堆点集&#xff0c;还有一个小的窗口&#xff0c;这个窗口可能是圆形的&#xff0c;现在你可能要移动这个窗口到点集密度最大的区域当中。 如下图&#xff1a; 最开始的窗口是蓝色圆环的区域&#xff0c;命名为C1…

数据结构之图:图的搜索,Python代码实现——23

图的搜索 深度优先搜索(Depth First Search) 定义 从例子出发理解 DFS是一种用于遍历或搜寻树类或图类数据结构的算法&#xff0c;这种算法从根结点出发&#xff08;如果是图&#xff0c;则任意选择一个顶点作为根结点&#xff09;&#xff0c;在回溯之前会尽可能地遍历每一…

人脸识别案例:【实战】opencv人脸检测+Haar特征分类器

1 基础 我们使用机器学习的方法完成人脸检测&#xff0c;首先需要大量的正样本图像&#xff08;面部图像&#xff09;和负样本图像&#xff08;不含面部的图像&#xff09;来训练分类器。我们需要从其中提取特征。下图中的 Haar 特征会被使用&#xff0c;就像我们的卷积核&…

数据结构之图:用图解决案例,Python代码实现——24

用图解决畅通工程案例与途径查找 代码中需要引入的类方法代码链接&#xff1a; 无向图Undigraph深度优先搜索DFS与广度优先搜索BFS 畅通工程-续 介绍 案例和之前并查集中实现的一样&#xff0c;但问题略有改动&#xff0c;需要判断9-10城市是否相通&#xff0c;9-8城市是否…

【在虚拟环境下完美解决】1698: error: (-215:Assertion failed) empty() in function cv::CascadeClassifier

问题描述 官方文档做的Demo发现遇到了错误提示如下&#xff1a; error: (-215:Assertion failed) !empty() in function ‘cv::CascadeClassifier::detectMultiScale’ 错误的原因&#xff1a; 出现 error: (-215:Assertion failed) !empty() in function ‘cv::CascadeClassif…

计算机视觉概述:视觉任务+场景领域+发展历程+典型任务

一、什么是计算机视觉 定义&#xff1a;计算机视觉&#xff08;Computer vision&#xff09;是⼀⻔研究如何使机器“看”的科学&#xff0c;更 进⼀步的说&#xff0c;就是指⽤摄影机和计算机代替⼈眼对⽬标进⾏识别、跟踪和测量 等&#xff0c;⽤计算机处理成为更适合⼈眼观察…

数据结构之图:有向图的介绍与实现,Python代码实现——25

有向图的介绍 引入 在实际生活中,很多应用相关的图都是有方向性的,最直观的就是网络,可以从A页面通过链接跳转到B页面,那么a和b连接的方向是a->b,但不能说是b->a,此时我们就需要使用有向图来解决这一类问题,它和我们之前学习的无向图,最大的区别就在于连接是具有方向的…

图像分类_01图像分类简介:挑战+近邻分类器+CIFAR-10数据集概述

2.1.1 图像分类 任务目的&#xff1a;对输入的图像赋予一个标签&#xff0c;这个标签在指定类别集合中。 下面这个例子中&#xff0c;图像分类模型拍摄一张图像并将概率分配给4个标签{cat&#xff0c;dog&#xff0c;hat&#xff0c;mug}。如图所示&#xff0c;请记住&#xf…

数据结构之图:有向图的拓扑排序,Python代码实现——26

有向图的拓扑排序 拓扑排序介绍 什么是拓扑排序&#xff1f; 一个有向图的拓扑排序&#xff08;Topological sort 或 Topological ordering&#xff09;是根据其有向边从顶点U到顶点V对其所有顶点的一个线性排序举个例子&#xff1a;让一个拓扑排序的图中的所有顶点代表某项…