您的位置:首页 > 房产 > 家装 > 开发一个微信小程序多少钱_清远市清城区发布_怎么做seo网站关键词优化_厦门seo优化多少钱

开发一个微信小程序多少钱_清远市清城区发布_怎么做seo网站关键词优化_厦门seo优化多少钱

2025/1/10 22:16:12 来源:https://blog.csdn.net/Menior_/article/details/144389114  浏览:    关键词:开发一个微信小程序多少钱_清远市清城区发布_怎么做seo网站关键词优化_厦门seo优化多少钱
开发一个微信小程序多少钱_清远市清城区发布_怎么做seo网站关键词优化_厦门seo优化多少钱

目录

一、红黑树的来历与目的

二、红黑树的定义

三、红黑树的特性

四、红黑树的存储结构

五、红黑树的节点旋转操作

5.1左旋操作

5.2右旋操作

六、红黑树的插入节点操作

七、红黑树的查找操作

八、红黑树的删除操作


一、红黑树的来历与目的

红黑树是一种自平衡的二叉查找树,是一种高效的查找树。它是由Rudolf Bayer于1972年发明,在当时被称为对称二叉B树。后来,在1978年被 Leo J. Guibas 和 Robert Sedgewick 修改为如今的红黑树。

红黑树的目的是实现比AVL树更为高效的插入与删除操作:

AVL树:适合应用于搜索场景,以查为主。

红黑树:适合用于频繁插入、删除场景,其实用性更加强。

而且红黑树相较于AVL树有更广泛的用途,红黑树具有良好的效率,它可在 O(logN) 时间内完成查找、增加、删除等操作。因此,红黑树在业界应用很广泛,比如 Java 中的 TreeMap,JDK 1.8 中的 HashMap、C++ STL 中的 map 均是基于红黑树结构实现的。

二、红黑树的定义

‌红黑树是一种自平衡的二叉查找树,主要用于实现关联数组。‌它通过特定的操作来保持树的平衡,从而确保在插入和删除操作后,树的性能不会显著下降。

定义红黑树的代码如下:

#include <iostream>
using namespace std;// 定义红黑树节点的颜色
enum Color {RED,BLACK
};// 红黑树的节点结构
struct Node {int data;bool color;Node* left;Node* right;Node* parent;Node(int data) : data(data), color(RED), left(NULL), right(NULL), parent(NULL) {}
};

三、红黑树的特性

对于一个红黑树必须满足以下的6个特性:

1.红黑树是一个二叉排序树

2.每个节点要么是红色,要么是黑色

3.根结点是黑色的

4.叶子节点(外部节点,NULL节点、失败的节点)都是黑色的

5.红色节点的父节点和子节点都是黑色的(不存在两个相邻的红色节点)

6.对于每一个节点,从该节点到任一叶子结点的路径上,其所含黑色节点的数量相同

简称为:左根右;根叶黑;不红红;黑路同。

左根右:表示红黑树满足 左子节点<根节点<右子节点,也就是满足排序条件

根叶黑:表示跟节点和叶子节点都是黑色的a

不红红:表示不能有两个连续的红色节点(父节点和子节点不可能同时是红色的)

黑路同:表示从任意应该节点走到子节点路径上的黑色节点数量是相同的

 四、红黑树的存储结构

红黑树的存储一般需要靠结构体来实现,用结构体实现的代码如下:

//宏定义颜色
#define red 0
#define black 1//数据类型Datatype
typedef char Datatype;
//红黑树节点存储结构
typedef struct node {Datatype data;int color;int key;//排序键值,根据key大小排序struct node* par;//父节点指针struct node* left, * right;//左右子节点指针
}Node;//红黑树的定义rbtree
typedef struct tree {Node* root;//指向根节点指针Node* nil;//叶子节点(哨兵)
}rbtree;

五、节点的旋转操作

红黑树与AVL树一样,都是二叉查找树;在这种数据结构中,旋转操作是一种很常规的做法;通常通过旋转的操作来使红黑树达到平衡的状态,旋转操作通常分为左旋和右旋。

5.1左旋操作

左旋是一种将某个节点的右子节点旋转上来的操作。也就是说当前节点的右子节点顶替了自己,然后自己变为右子节点的左子节点,以保持树的平衡。

代码实现如下:

#include <iostream>
using namespace std;// 定义红黑树节点的颜色
enum Color {RED,BLACK
};// 红黑树的节点结构
struct Node {int data;bool color;Node* left;Node* right;Node* parent;Node(int data) : data(data), color(RED), left(NULL), right(NULL), parent(NULL) {}
};class RedBlackTree {
private:Node* root;
public:RedBlackTree() : root(NULL) {}// 左旋操作void leftRotate(Node* x) {if (x == NULL) {// 检查输入节点是否为空return; }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;} else if (x == x->parent->left) {x->parent->left = y;} else {x->parent->right = y;}y->left = x;x->parent = y;}
};int main() {RedBlackTree rbt;Node* root = new Node(10);Node* node1 = new Node(20);root->right = node1;node1->parent = root;rbt.leftRotate(root);return 0;
}

5.2右旋操作

同样的右旋也是将左子节点顶替自己成为父节点, 然后自己成为左子节点的右子节点。

代码实现如下:

#include <iostream>
using namespace std;// 定义红黑树节点的颜色
enum Color {RED,BLACK
};// 红黑树的节点结构
struct Node {int data;bool color;Node* left;Node* right;Node* parent;Node(int data) : data(data), color(RED), left(NULL), right(NULL), parent(NULL) {}
};class RedBlackTree {
private:Node* root;
public:RedBlackTree() : root(NULL) {}// 右旋操作void rightRotate(Node* y) {if (y == NULL) {// 检查输入节点是否为空return; }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->right) {y->parent->right = x;} else {y->parent->left = x;}x->right = y;y->parent = x;}
};int main() {RedBlackTree rbt;Node* root = new Node(10);Node* node1 = new Node(5);root->left = node1;node1->parent = root;rbt.rightRotate(root);return 0;
}

六、红黑树的插入节点操作

红黑树的插入操作分两步走:

找到插入位置
进行自平衡调整
注意:插入节点初始为红色

原因分析:因为红黑树中任意一个节点到叶子节点路径所含黑色节点数量相同,也就是说如果我插入的节点为黑色的话,那么就会破坏红黑树的要求,所以插入的节点必须是红色节点,才能保证红黑树的性质。

下面就开始讨论红黑树的几种插入情况:

1.插入的是空树
这是最简单的插入情况,当插入第一个节点的时候,红黑树为空我们只需要让根节点指向这个节点即可。操作如下:

根节点指向此节点
把根节点染黑


2.插入节点的key重新重复
这种情况的话我们可以根据自己喜好去处理,如果出现了重复的key,那么就把这个key里面的值进行更新;或者我们不进行插入操作,因为key不可以重复,直接退出插入操作。

3.插入节点的父节点是黑色
这很好处理,直接插入就行了,因为父节点为黑色,插入节点为红色,所以不会影响红黑树的平衡性。直接插入即可。


4.插入节点的父节点是红色
这种情况是最为复杂的,由于父节点颜色是红色,所以要进行平衡调整,所以要去进一步的讨论才行。那具体根据什么去调整呢?是看叔叔节点的颜色来调整(父节点的兄弟节点),具体分以下几种情况:

大的有两种情况,要看父节点是祖父节点的左边还是右边,下面我就以父节点为左子节点为例子:

 下文图标说明:

t 表示插入的节点

P表示父节点

B表示叔叔节点

PP表示祖父节点

4.1:父节点是祖父节点的左子节点
4.1.1叔叔节点是红色 

如果叔叔节点的颜色是红色的话,这里不需要进行旋转操作,只需要让父节点和叔叔节点颜色变为黑色,祖父节点颜色变为红色即可。流程如下:

把P 和B 节点染黑
把PP节点染红

 4.1.2叔叔节点是黑色
这里的话又要去分两种情况:

插入节点是父节点的左子节点
插入节点是父节点的右子节点


4.1.2-1 插入节点是作左子节点
 如果插入的节点是父节点的左子节点的话,那么要进行以下操作:

把P染黑
把PP染红
对PP进行右旋

4.1.2-2插入节点是作右子节点
 如果插入节点是作为父节点的右子节点的话,要进行以下操作:

对P进行左旋
把t 染黑
把PP染红
对PP进行右旋

 4.2:父节点是祖父节点的右子节点

这里要分成三种情况来讨论:

4.2.1叔叔节点是红色
操作步骤如下:

把B(叔叔节点)和P(父节点)然黑
把PP(祖父节点)染红

4.2.2:  叔叔节点是黑色
同样的也是分以下两种情况讨论: 

4.2.1-1 插入节点是作左子节点
 对P 进行右旋
将t 染黑
将PP 然红
对PP 进行左旋


​4.2.1-2 插入节点是作右子节点
 将P 染黑
将PP 然红
对PP进行左旋

附上代码:

#include <iostream>
using namespace std;// 定义红黑树节点的颜色
enum Color {RED,BLACK
};// 红黑树的节点结构
struct Node {int data;bool color;Node* left;Node* right;Node* parent;Node(int data) : data(data), color(RED), left(NULL), right(NULL), parent(NULL) {}
};class RedBlackTree {
private:Node* root;// 左旋操作void leftRotate(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;} else if (x == x->parent->left) {x->parent->left = y;} else {x->parent->right = y;}y->left = x;x->parent = y;}// 右旋操作void rightRotate(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->right) {y->parent->right = x;} else {y->parent->left = x;}x->right = y;y->parent = x;}// 辅助函数:获取节点的颜色,如果节点为空,默认为黑色Color getColor(Node* node) {if (node == NULL) {return BLACK;}return node->color;}// 辅助函数:插入节点的修复操作,确保红黑树的性质void fixInsert(Node* z) {Node* parent = NULL;Node* grandparent = NULL;while ((z!= root) && (getColor(z) == RED) && (getColor(z->parent) == RED)) {parent = z->parent;grandparent = parent->parent;if (parent == grandparent->left) {Node* uncle = grandparent->right;if (getColor(uncle) == RED) {grandparent->color = RED;parent->color = BLACK;uncle->color = BLACK;z = grandparent;} else {if (z == parent->right) {z = parent;leftRotate(z);parent = z->parent;}parent->color = BLACK;grandparent->color = RED;rightRotate(grandparent);}} else {Node* uncle = grandparent->left;if (getColor(uncle) == RED) {grandparent->color = RED;parent->color = BLACK;uncle->color = BLACK;z = grandparent;} else {if (z == parent->left) {z = parent;rightRotate(z);parent = z->parent;}parent->color = BLACK;grandparent->color = RED;leftRotate(grandparent);}}}root->color = BLACK;}public:RedBlackTree() : root(NULL) {}// 插入节点操作void insert(int data) {Node* z = new Node(data);Node* y = NULL;Node* x = root;while (x!= NULL) {y = x;if (z->data < x->data) {x = x->left;} else {x = x->right;}}z->parent = y;if (y == NULL) {root = z;} else if (z->data < y->data) {y->left = z;} else {y->right = z;}fixInsert(z);}// 中序遍历void inorderTraversal(Node* node) {if (node == NULL) {return;}inorderTraversal(node->left);cout << node->data << " ";inorderTraversal(node->right);}// 打印中序遍历结果void printInorder() {inorderTraversal(root);cout << endl;}
};int main() {RedBlackTree rbt;rbt.insert(10);rbt.insert(20);rbt.insert(30);rbt.insert(5);rbt.insert(15);rbt.insert(25);rbt.printInorder();return 0;
}

七、红黑树的查找操作

在查找操作方面,红黑树的查找与AVL树中的操作基本一致;都是通过key值从左向右查找,找到就返回即可。

//根据key查找
Node* Search_key(rbtree* T, int target) {assert(T);assert(T->root);Node* cur = T->root;while (cur) {if (cur->key == target)return cur;//找到就返回else if (cur->key > target)cur = cur->left;elsecur = cur->right;}printf("The target is not exist\n");return NULL;
}

八、红黑树的删除操作

红黑树的删除所有情况如下所示:

删除的是叶子节点(下面又分2种情况)
删除节点的颜色是红色
删除节点的颜色是黑色(下面再分5种情况)
兄弟节点没有左右孩子
兄弟节点左孩子为红色,右孩子为黑色
兄弟节点右孩子为红色,左孩子为黑色
兄弟节点有左右孩子,且都为红色
兄弟节点有左右孩子,且都为黑色(兄弟节点为红色)
删除的只有左子节点,没有右子节点
删除的只有右子节点,没有左子节点
删除的既有左子节点,又有右子节点

根据以上情况,用代码实现的操作如下:

// 删除操作的修复函数void fixDelete(Node* x) {while (x!= root && getColor(x) == BLACK) {if (x == x->parent->left) {Node* w = x->parent->right;if (getColor(w) == RED) {w->color = BLACK;x->parent->color = RED;leftRotate(x->parent);w = x->parent->right;}if (getColor(w->left) == BLACK && getColor(w->right) == BLACK) {w->color = RED;x = x->parent;} else {if (getColor(w->right) == BLACK) {w->left->color = BLACK;w->color = RED;rightRotate(w);w = x->parent->right;}w->color = x->parent->color;x->parent->color = BLACK;w->right->color = BLACK;leftRotate(x->parent);x = root;}} else {Node* w = x->parent->left;if (getColor(w) == RED) {w->color = BLACK;x->parent->color = RED;rightRotate(x->parent);w = x->parent->left;}if (getColor(w->right) == BLACK && getColor(w->left) == BLACK) {w->color = RED;x = x->parent;} else {if (getColor(w->left) == BLACK) {w->right->color = BLACK;w->color = RED;leftRotate(w);w = x->parent->left;}w->color = x->parent->color;x->parent->color = BLACK;w->left->color = BLACK;rightRotate(x->parent);x = root;}}}x->color = BLACK;}public:RedBlackTree() : root(NULL) {}// 删除节点操作void deleteNode(int data) {Node* z = root;Node* x, * y;while (z!= NULL) {if (z->data == data) {break;} else if (z->data < data) {z = z->right;} else {z = z->left;}}if (z == NULL) {return;}y = z;Color yOriginalColor = y->color;if (z->left == NULL) {x = z->right;transplant(z, z->right);} else if (z->right == NULL) {x = z->left;transplant(z, z->left);} else {y = minimum(z->right);yOriginalColor = y->color;x = y->right;if (y->parent == z) {if (x!= NULL) {x->parent = y;}} else {transplant(y, y->right);y->right = z->right;y->right->parent = y;}transplant(z, y);y->left = z->left;y->left->parent = y;y->color = z->color;}if (yOriginalColor == BLACK) {fixDelete(x);}}

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com