一,红黑树简介
1,红黑树的特点
首先我们得知道红黑树是什么才能知道这棵树里面应该有啥,红黑树其实有以下几个特点:
1,红黑树的每个节点是黑色或者红色的。
2,红黑树的根节点是黑色的。
3,红黑树不能出现两个连续的红节点。
4,红黑树的最长路径的长度不超过最短路径的长度的两倍。
5,红黑树的叶子节点是黑色的(这里的叶子节点指的是最后面的空节点)。
如下便是一棵红黑树:这里的规则五指的叶子节点便是最后的NIL节点。
2,红黑树的节点
在知道了红黑树的特点以后,便可以来讲讲红黑树节点的节点该如何构建了。按照上图红黑树节点成员有以下几个:
1,_left指针 。
2,_right指针 。
3,_parent指针(主要是为了构建三叉戟方便找到父节点)。
4,表示颜色的变量(enum colour,枚举类型的colour)。
5,表示节点值的变量_val。
在知道上面的节点成员以后便可以开始构建节点了,构建出来的节点如下:
enum Colour //枚举类型的颜色。 {RED,BLACK}; template<class K,class V> struct RBNode { RBNode(const pair<K,V>&val)//为了后面插入值做准备。 :_val(val) ,_colour(RED)//一般插入的节点默认都是红色节点 {} //成员 pair<K, V>_val; RBNode<K, V>* _right; RBNode<K, V>* _left; RBNode<K, V>* parent; Colour _colour; };
3,红黑树的插入
红黑树归根结底还是一棵搜索二叉树,所以红黑树还是要符合搜索二叉树大的在右,小的在左的特点。所以红黑树的插入部分在前面还是和二叉·搜索树前面是一样的:
typedef RBNode<K, V> Node; bool Insert(const pair<K, V>& kv) { if (_root == nullptr)//当根节点为NULL时,要让他变成一个节点 { _root = new Node(val); _root->_colour = BLACK;//根节点为黑色 return true; } Node cur = _root; Node parent = nullptr; while (cur) { if (kv->first <cur._kv->first ) { parent = cur; cur = cur->_left } else if (kv->first > cur._kv->first) { parent = cur; cur = cur->_right; } else//插入一个已经存在的值 { return false; } } //cur走到空,parent时cur的上一个节点 //生成一个cur节点 cur = new Node(kv); //判断应该在parent的左边还是右边 if (cur._kv->first < parent._kv->first) { parent->_left = cur; } else { parent->_right = cur; } //挂好节点,以上的操作便是和二叉搜索树一样的操作 }
然后便是不一样的,独属于红黑树的调整操作了。首先要明确一个点,红黑树在啥时候要调整呢?当然是在出现两个连续红色节点的情况下。
调整的方式有两种:1,直接变色。
2,旋转+变色。
并且旋转+变色的情况又分为四种:左,右,左右,右左。然后变色。(其实旋转的目的就是为了将下面四种情况转化为第一种情况来变色)。
在调整之前,我们来将这些情况的总体示意图画出如下,并将几个关键节点(grandparent,parent,uncle,cur)画图如下:
到时候判断如何调整便要看这些节点的关系。现在来讨论五种情况:
1,变色
只变色的情况如下:1,parent为red, 2,uncle为red。
示意图如下:
此时只需要变色,变色步骤如下:g变红,p和u变黑。
代码操作如下:
if (uncle->_colour == RED)//只变色 { grandparent->_colour = RED; uncle->_colour = BLACK; parent->_colour = BLACK; //向上调整,因为grandparent节点的颜色原来为BLACK,现在改为RED。可能会发生冲突情况,所以要向上调整。 cur = grandparent; parent = cur->_parent; }
2,旋转加变色
旋转+变色的情况分为四种
1.左+变色
要发生这个情况的调整的情况如下:
1.parent是grandparent的右边,并且cur是parent的右边。
示意图如下:
在这种情况下便要发生以grandparent为中心的左旋:
然后再将cur变为黑色。如下:
代码如下:
if (parent == grandparent->_right) { if (cur == parent->_right) { RevolveL(grandparent);//旋转+变色 parent->_colour = BLACK; grandparent->_colour = RED; break; } }
2,右左+变色
要发生这样调整的情况:parent在grandparent的右边,cur在parent的左边。
示意图如下:
要调整这种情况,首先得以parent为中心右旋:变成左边高
然后再以grandparent为中心左旋:
然后变色:cur变黑,grandparent变红。
代码:
else if(cur == parent->_left) { RevolveR(parent);//先右旋 RevolveL(grandparent); //变色 cur->_colour = BLACK; grandparent->_colour = RED; break; }
3,右+变色
发生这种变色的情况如下:parent在grandparent的左边,cur在parent的左边。
如下图所示:
在这种情况下我要先右旋:
然后再变色:parent的颜色变黑,grandparent的颜色变红
代码:
if (cur == parent->_left) { RevolveR(grandparent); parent->_colour = BLACK; grandparent->_colour = RED; break; }
4,右左+变色
发生这种调整的情况是:parent在grandparent的左边,cur在parent的右边。如下图:
调整策略如下:
1.先以parent为中心左旋:
再以grandparent为中心向右旋转:
然后再变色:
代码如下:
else if(cur == parent->_right) { RevolveL(parent); RevolveR(grandparent); grandparent->_colour = RED; cur->_colour = BLACK; break; }
调整的整体代码如下:while (parent && parent->_colour == RED) { if (parent == _root) { break; } //找叔叔 if (parent == grandparent->_left) { uncle = grandparent->_right; } else { uncle = grandparent->_left; } if (uncle&&uncle->_colour == RED)//只变色 { grandparent->_colour = RED; uncle->_colour = BLACK; parent->_colour = BLACK; //向上调整,因为grandparent节点的颜色原来为BLACK,现在改为RED。可能会发生冲突情况,所以要向上调整。 cur = grandparent; parent = cur->_parent; } else if (uncle == nullptr || uncle->_colour == BLACK) { if (parent == grandparent->_right) { if (cur == parent->_right) { RevolveL(grandparent); parent->_colour = BLACK; grandparent->_colour = RED; break; } else if(cur == parent->_left) { RevolveR(parent);//先右旋 RevolveL(grandparent); //变色 cur->_colour = BLACK; grandparent->_colour = RED; break; } } else if (parent == grandparent->_left) { if (cur == parent->_left) { RevolveR(grandparent); parent->_colour = BLACK; grandparent->_colour = RED; break; } else if(cur == parent->_right) { RevolveL(parent); RevolveR(grandparent); cur->_colour = BLACK; grandparent->_colour = RED; break; } } }
4,平衡检查
1,有没有连续的红节点出现。
2,每条路径上的黑色节点的个数是否相同。
检查思路如下:
1.当遇到红节点时便去查看一下他的父亲节点是否为红。
2.检查黑色节点的个数便要传入一个参考值,refblacknum,然后检查每一条路径上的黑色节点个数是否等于这个值。
代码如下:
检查平衡的函数:
bool isBlance() { if (_root == nullptr) { return true; } int blackNum = 0; Node* cur = _root; while (cur) { if (cur->_colour == BLACK) { blackNum++; } cur = cur->_left; } return check(_root, 0, blackNum); }
check函数如下:
bool check(Node* root, int Num, int refBlackNum) { if (root == nullptr) { if (Num != refBlackNum) { cout << "黑色节点个数不同:"; cout << _root->_kv.first << endl; return false; } return true; } if (root->_colour == RED) { if (root->_parent->_colour == RED) { cout << root->_kv.first << " " << root->_parent->_kv.first << endl; printf("出现两个连续的红节点\n"); return false; cout << root->_kv.first << endl; } } if (root->_colour == BLACK) { Num++; } return check(root->_left, Num, refBlackNum) && check(root->_right, Num, refBlackNum); }