#ifndef __RED_BLACK_TREE_H__
#define __RED_BLACK_TREE_H__ namespace lxf {
template <typename Key, typename Value>
class RedBlackTree {
struct Node {
enum Color { RED, BLACK };
Key key;
Value value;
Node *left;
Node *right;
Node *parent;
Color color;
Node(Key pkey, Value pvalue, Color pcolor) : key(pkey), value(pvalue), left(nullptr), right(nullptr), parent(nullptr), color(pcolor) {}
};
Node *root;
void rotateLeft(Node *x);
void rotateRight(Node *x); Node* deleteMin(Node* head);
Node* min(Node *head);
public:
void insert(Key key, Value value);
void erase(Key key);
RedBlackTree():root(nullptr) {}
}; /*
* 对红黑树的节点(x)进行左旋转
*
* 左旋示意图(对节点x进行左旋):
* px px
* / /
* x y
* / \ --(左旋)--> / \ #
* lx y x ry
* / \ / \
* ly ry lx ly
*
*
*/
template<typename Key, typename Value>
inline void RedBlackTree<Key, Value>::rotateLeft(Node * x)
{
Node *y = x->right; x->right = y->left;
if (y->left != nullptr)
y->left->parent = x; y->parent = x->parent;
if (x->parent == nullptr)
root = y;
else {
if (x->parent->left == x)
x->parent->left = y;
else
x->parent->right = y;
}
y->left = x;
x->parent = y;
} /*
* 对红黑树的节点(y)进行右旋转
*
* 右旋示意图(对节点y进行左旋):
* py py
* / /
* y x
* / \ --(右旋)--> / \ #
* x ry lx y
* / \ / \ #
* lx rx rx ry
*
*/
template<typename Key, typename Value>
inline void RedBlackTree<Key, Value>::rotateRight(Node * x)
{
Node *x = y->left;
y->left = x->right;
if (x->right != nullptr)
x->right->parent = y;
x->parent = y->parent;
if (y->parent == nullptr)
root = x;
else {
if (y == y->parent->right)
y->parent->right = x;
else
y->parent->left = x;
}
x->right = y;
y->parent = x;
} template<typename Key, typename Value>
inline typename RedBlackTree<Key, Value>::Node * RedBlackTree<Key, Value>::deleteMin(Node * head)
{
if (head == nullptr)
return nullptr;
Node *node = head;
Node *lastNode = nullptr;
while (node->left != nullptr)
{
lastNode = node;
node = node->left;
}
lastNode->left = node->right;
if (node->right != nullptr)
node->right->parent = lastNode;
return head;
} template<typename Key, typename Value>
inline typename RedBlackTree<Key, Value>::Node * RedBlackTree<Key, Value>::min(Node * head)
{
if (head == nullptr)
return nullptr;
Node *node = head;
while (node->left != nullptr)
node = node->left;
return node;
} template<typename Key, typename Value>
inline void RedBlackTree<Key, Value>::insert(Key key, Value value)
{
Node *parent = nullptr;
Node **ppNode = &root;
while (*ppNode != nullptr)
{
parent = *ppNode;
if (key < (*ppNode)->key)
ppNode = &((*ppNode)->left);
else
ppNode = &((*ppNode)->right);
}
if (*ppNode == nullptr) {
*ppNode = new Node(key, value, RED);
(*ppNode)->parent = parent;
} // 结点是根
if (parent == nullptr) {
root->color = Node::BLACK;
return;
} Node *node = *ppNode;
// 调整
while (node->parent->color == Node::RED)
{
Node *parent = node->parent;
Node *gparent = node->parent->parent;
if (parent == gparent->left)
{
Node *uncle = gparent->right;
if (uncle->color == RED)
{
parent->color = BLACK;
uncle->color = BLACK;
gparent->color = RED;
node = gparent;
}
else if (uncle->color == BLACK)
{
if (node == parent->right)
{
node = parent;
rotateLeft(node);
}
else
{
parent->color = BLACK;
gparent->color = RED;
rotateRight(gparent);
}
}
}
else if (parent == gparent->right)
{
Node *uncle = gparent->left;
if (uncle->color == RED)
{
parent->color = BLACK;
uncle->color = BLACK;
gparent->color = RED;
node = gparent;
}
else if (uncle->color == BLACK)
{
if (node == parent->left)
{
node = parent;
rotateRight(node);
}
else
{
parent->color = BLACK;
gparent->color = RED;
rotateLeft(gparent);
}
}
}
}
} template<typename Key, typename Value>
inline void RedBlackTree<Key, Value>::erase(Key key)
{
Node *lastNode = nullptr;
Node *node = root;
while (node != nullptr)
{
if (key < node->key) {
lastNode = node;
node = node->left;
}
else if (key > node->key) {
lastNode = node;
node = node->right;
}
else {
Node **plastNode = nullptr;
// 注意树根
if (lastNode == nullptr)
plastNode = &root;
else
plastNode = &lastNode;
// 无节点的情况
if (node->left == nullptr && node->right == nullptr)
{
if ((*plastNode)->left == node)
{
(*plastNode)->left = nullptr;
delete node;
}
else if ((*plastNode)->right == node)
{
(*plastNode)->right = nullptr;
delete node;
}
}
// 两个节点的情况
else if (node->left != nullptr && node->right != nullptr) {
Node *star = min(node->right);
star->right = deleteMin(node->right);
if (star->right != nullptr)
star->right->parent = star;
star->left = node->left;
if (star->left != nullptr)
star->left->parent = star;
delete node;
}
// 只有一个节点的情况
else if (node->left == nullptr) {
if ((*plastNode)->left == node) {
(*plastNode)->left = node->right;
if (node->right != nullptr)
node->right->parent = *plastNode;
delete node;
}
else if ((*plastNode)->right == node) {
(*plastNode)->right = node->right;
if (node->right != nullptr)
node->right->parent = *plastNode;
delete node;
}
}
else if (node->right == nullptr) {
if ((*plastNode)->right == node) {
(*plastNode)->right = node->left;
if (node->left != nullptr)
node->left->parent = *plastNode;
delete node;
}
else if ((*plastNode)->left == node) {
(*plastNode)->left = node->left;
if (node->left != nullptr)
node->left->parent = *plastNode;
delete node;
}
} }
} while (node != root && node->color == BLACK)
{
if (node == node->parent->left)
{
Node* brother = node->parent->right;
if (brother->color == RED)
{
brother->color = BLACK;
node->parent->color = RED;
rotateLeft(node->parent);
}
else
{
if (brother->left->color == BLACK && brother->right->color == BLACK)
{
brother->color = RED;
node = node->parent;
}
else if (brother->right->color == BLACK)
{
brother->color = RED;
brother->left->color = BLACK;
RotateRight(brother);
}
else if (brother->right->color == RED)
{
brother->color = node->parent->color;
node->parent->color = BLACK;
brother->right->color = BLACK;
rotateLeft(node->parent);
node = root;
}
}
}
else
{
Node* brother = node->parent->left;
if (brother->color == RED)
{
brother->color = BLACK;
node->parent->color = RED;
rotateRight(node->parent);
}
else
{
if (brother->left->color == BLACK && brother->right->color == BLACK)
{
brother->color = RED;
node = node->parent;
}
else if (brother->left->color == BLACK)
{
brother->color = RED;
brother->right->color = BLACK;
rotateLeft(brother);
}
else if (brother->left->color == RED)
{
brother->color = node->parent->color;
node->parent->color = BLACK;
brother->left->color = BLACK;
rotateRight(node->parent);
node = root;
}
}
}
}
node->parent = root; //最后将node置为根结点,
node->color = BLACK; //并改为黑色。
}
} #endif /*__RED_BLACK_TREE_H__*/

红黑树 C++实现的更多相关文章

  1. 红黑树——算法导论(15)

    1. 什么是红黑树 (1) 简介     上一篇我们介绍了基本动态集合操作时间复杂度均为O(h)的二叉搜索树.但遗憾的是,只有当二叉搜索树高度较低时,这些集合操作才会较快:即当树的高度较高(甚至一种极 ...

  2. jdk源码分析红黑树——插入篇

    红黑树是自平衡的排序树,自平衡的优点是减少遍历的节点,所以效率会高.如果是非平衡的二叉树,当顺序或逆序插入的时候,查找动作很可能会遍历n个节点 红黑树的规则很容易理解,但是维护这个规则难. 一.规则 ...

  3. 谈c++ pb_ds库(二) 红黑树大法好

    厉害了,没想到翻翻pb_ds库看到这么多好东西,封装好的.现成的splay.红黑树.avl... 即使不能在考场上使用也可以用来对拍哦 声明/头文件 #include <ext/pb_ds/tr ...

  4. 定时器管理:nginx的红黑树和libevent的堆

    libevent 发生超时后, while循环一次从堆顶del timer——直到最新调整的最小堆顶不是超时事件为止,(实际是del event),但是会稍后把这个timeout的 event放到ac ...

  5. 从2-3-4树到红黑树(下) Java与C的实现

    欢迎探讨,如有错误敬请指正 如需转载,请注明出处   http://www.cnblogs.com/nullzx/ 相关博客: 从2-3-4树到红黑树(上) 从2-3-4树到红黑树(中) 1. 实现技 ...

  6. 红黑树/B+树/AVL树

    RB Tree 红黑树  :http://blog.csdn.net/very_2/article/details/5722682 Nginx的RBTree实现   :http://blog.csdn ...

  7. 论AVL树与红黑树

    首先讲解一下AVL树: 例如,我们要输入这样一串数字,10,9,8,7,15,20这样一串数字来建立AVL树 1,首先输入10,得到一个根结点10 2,然后输入9, 得到10这个根结点一个左孩子结点9 ...

  8. DataStructure——红黑树学习笔记

    1.前言 本文伪码和解释参考: http://blog.csdn.net/v_JULY_v/article/details/6105630 C实现的源码本文未贴出,请见: http://blog.cs ...

  9. 红黑树(Red-Black tree)

    红黑树又称红-黑二叉树,它首先是一颗二叉树,它具体二叉树所有的特性.同时红黑树更是一颗自平衡的排序二叉树.我们知道一颗基本的二叉树他们都需要满足一个基本性质–即树中的任何节点的值大于它的左子节点,且小 ...

  10. map,hash_map, hash_table, 红黑树 的原理和使用

    在刷算法题的时候总是碰到好多题,号称可以用hash table来解题.然后就蒙圈了. 1.首先,map和hash_map的区别和使用: (1)map底层用红黑树实现,hash_map底层用hash_t ...

随机推荐

  1. MySQL5.6自动化部署(二进制)

    ###### 二进制自动安装数据库脚本root密码MANAGER将脚本和安装包放在/root目录即可############### ######数据库目录/usr/local/mysql####### ...

  2. ASP.NET MVC中使用异步控制器

    线程池 一直想把项目改写成异步,但是ASP.NETMVC3下写的过于繁琐,.NET 4.5与ASP.NET MVC下代码写起来就比较简单了, MS好像也一直喜欢这样搞,每一个成熟的东西,都要演变好几个 ...

  3. 把记事本文件固定在Win8的开始屏幕

    1.创建该文件的桌面快捷方式: 2.将快捷方式拷贝至开始菜单目录,在开始屏幕的查看全部中可以看见该文件快捷: 3.在查看全部中右键点击该快捷,选择固定在开始屏幕:

  4. JSP编译为Java类

    JSP编译为Java类: 注意可以随便写import的内容:可以写类属性.方法.main函数.内部类:可以使用内部类: JSP: <%@ page language="java&quo ...

  5. Debian8 安装wordpress博客

    首先要安装 php5-gd 已保证wordpress可以正常安装 下载Wordpress wget http://wordpress.org/latest.tar.gz 解压 tar -xzvf la ...

  6. Visual Studio项目模板与向导开发

    在[Xamarin+Prism开发详解系列]里面经常使用到[Prism unity app]的模板创建Prism.Forms项目: 备注:由于Unity社区已经不怎么活跃,下一个版本将会有Autofa ...

  7. [Echarts]用Echarts绘制饼状图

    在项目网站的网页中,有这样一幅图: 心血来潮,想使用百度Echarts来绘制一下,可是没能绘制得完全一样,Echarts饼状图的label不能在图形下面放成一行,最后的效果是这样子的: 鼠标移动到it ...

  8. .Net程序员学用Oracle系列(12):增删改查

    1.插入语句 1.1.INSERT 1.2.INSERT ALL 2.删除语句 2.1.DELETE 2.2.TRUNCATE 3.更新语句 3.1.UPDATE 3.2.带子查询的 UPDATE 3 ...

  9. windows下composer安装不了或composer命令无效的解决办法

    安装方法: http://docs.phpcomposer.com/00-intro.html#Installation-Windows (官方中文帮助文档) 问题描述1: 通过Composer-Se ...

  10. Syslog4j如何实现接收日志

    1.Syslog4j是一个实现Syslog(RFC3164)协议的Java开源类库包括客户端与服务器端.通过 UDP/IP, TCP/IP, TCP/IPover SSL/TLS, Unix Sysl ...