一、模拟实现map和set
1.1三个类模板的设计
template <class Key, class Value, class KeyOfValue, class Compare,class Alloc = alloc>class rb_tree{};//红黑树类模板template <class Key, class T, class Compare = less<Key>, class Alloc = alloc>
class map{};//map对红黑树进行了封装
template <class Key, class Compare = less<Key>, class Alloc = alloc>
class set{};//set对红黑树进行了封装
实际上红黑树类模板的第二个模板参数决定了红黑树的节点用来存放key还是pair;第一个模板参数是因为find之类的函数需要传递key,而第二个模板参数可能是pair无法实现实现直接使用key进行比较;第三个模板参数是为了解决insert是需要根据键值来进行搜索二叉树的建立,但是传递的参数是data,可能是键值,也可能是pair,所以需要使用仿函数,将key取出来,主要是为了map设计的;
为什么不使用仿函数直接比较而是返回key值,因为仿函数是最外层进行传入的,最外层传递比较方式,而内层的红黑树关心的是键值比较,所以红黑树使用KeyOfT;
1.2迭代器的设计
在rbtree中设计。
1.2.1begin/end
begin返回值是最左节点,end返回的是nullptr;
而库里面实现的是增加了一个哨兵位,使得哨兵位的左指向最小值,右指向最大值,end是哨兵位;好处是获取begin较快,但是删除和插入的成本增加了;
1.2.2++和- -的设计
++走中序遍历;
1.右不为空,访问右树的最左节点;
2.右为空,访问满足孩子是父亲的左孩子的最近祖先;
- - 反的走中序遍历
1.左不为空,访问左树的最右节点;
2.左为空,访问满足孩子是父亲的右孩子的最近祖先;
1.2.3const迭代器
RBTree通过模板参数实现const的迭代器类,set封装使用const迭代器来定义普通和const迭代器,即set使用的迭代器都是const迭代器;map使用pair<const K,V>来实现普通的迭代器不可以访问key;
1.2.4反向迭代器
1.3插入实现
注意返回值是pair,要注意对于set返回值要使用const迭代器的构造(普通迭代器对const迭代器的构造),因为普通迭代器和const迭代器会发生冲突;
对于tree迭代器类的设计其实是设计了一个迭代器的拷贝构造函数,set调用时变成普通迭代器对const迭代器的构造;然后tree迭代器内嵌了普通迭代器的类型。对于内嵌类型类外使用时不会给提示的,还必须typename修饰。
1.4operator[]
使用insert实现。
1.5map,tree,set的代码实现
namespace RbTree
{enum Colour{RED,BLACK,};template <class T>struct RBTreeNode{RBTreeNode(const T &data) : data_(data), left_(nullptr), right_(nullptr), parent_(nullptr), col_(RED) {}T data_;RBTreeNode<T> *left_;RBTreeNode<T> *right_;RBTreeNode<T> *parent_;Colour col_;};template <class T, class Ptr, class Ref>struct __TreeIterator{typedef RBTreeNode<T> node;typedef __TreeIterator<T, Ptr, Ref> Self;typedef __TreeIterator<T, T *, T &> iterator;__TreeIterator(node *node = nullptr) : node_(node) {}__TreeIterator(const iterator &it) : node_(nullptr){node_ = it.node_;}Ref operator*(){return node_->data_;}Ptr operator->(){return &(node_->data_);}Self &operator++(){if (node_->right_){// 右不为空,访问右树的最左节点node *subleft = node_->right_;while (subleft->left_){subleft = subleft->left_;}node_ = subleft;}else{// 右为空,访问满足孩子是父亲的左孩子的最近的祖先node *parent = node_->parent_;node *cur = node_;while (parent && cur != parent->left_){cur = parent;parent = parent->parent_;}// 判断是否找到祖先节点if (!parent){node_ = nullptr;}else{node_ = parent;}}return *this;}Self &operator--(){ // 右子树 根 左子树if (node_->left_){node *rightmost = node_->left_;while (rightmost->right_){rightmost = rightmost->right_;}node_ = rightmost;}else{node *parent = node_->parent_;node *cur = node_;while (parent && cur != parent->right_){cur = parent;parent = parent->parent_;}// 判断是否找到祖先节点if (!parent){node_ = nullptr;}else{node_ = parent;}}return *this;}bool operator!=(const Self &iterator) const{return node_ != iterator.node_;}bool operator==(const Self &iterator) const{return node_ == iterator.node_;}node *node_;};template <class K, class T, class KeyOfT>class RBTree{typedef RBTreeNode<T> node;public:typedef __TreeIterator<T, T *, T &> iterator;typedef __TreeIterator<T, const T *, const T &> const_iterator;iterator begin(){node *leftmost = root_;while (leftmost && leftmost->left_){leftmost = leftmost->left_;}return leftmost;}iterator end(){return nullptr;}const_iterator begin() const{node *leftmost = root_;while (leftmost && leftmost->left_){leftmost = leftmost->left_;}return leftmost;}const_iterator end() const{return nullptr;}node *find(const K &key){KeyOfT kot;node *cur = root_;while (cur){if (kot(cur->data_) < key){cur = cur->right_;}else if (kot(cur->data_) > key){cur = cur->left_;}else{return cur;}}return nullptr;}std::pair<iterator, bool> insert(const T &data);void rotatel(node *parent);void rotater(node *parent);void rotaterl(node *parent);void rotatelr(node *parent);void inorder(){_inorder(root_);}bool isrbtree(){return _isrbtree(root_);}bool checkcolour(node *root, int blacknum, int reference);private:void _inorder(node *root);bool _isrbtree(node *root);node *root_ = nullptr;};template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotatel(node *parent){node *cur = parent->right_;node *curleft = cur->left_;parent->right_ = curleft;node *ppnode = parent->parent_;if (curleft)curleft->parent_ = parent;parent->parent_ = cur;cur->left_ = parent;if (parent == root_){root_ = cur;cur->parent_ = nullptr;}else{if (ppnode->left_ == parent){ppnode->left_ = cur;}else{ppnode->right_ = cur;}cur->parent_ = ppnode;}}template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotater(node *parent){node *cur = parent->left_;node *curright = cur->right_;node *ppnode = parent->parent_;parent->left_ = curright;cur->right_ = parent;if (curright)curright->parent_ = parent;parent->parent_ = cur;if (parent == root_){root_ = cur;cur->parent_ = nullptr;}else{if (ppnode->left_ == parent){ppnode->left_ = cur;}else{ppnode->right_ = cur;}cur->parent_ = ppnode;}}template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotaterl(node *parent){rotater(parent->right_);rotatel(parent);}template <class K, class T, class KeyOfT>void RBTree<K, T, KeyOfT>::rotatelr(node *parent){rotatel(parent->left_);rotater(parent);}template <class K, class T, class KeyOfT>pair<typename RBTree<K, T, KeyOfT>::iterator, bool> RBTree<K, T, KeyOfT>::insert(const T &data){if (root_ == nullptr){root_ = new node(data);root_->col_ = BLACK;return {root_, true};}node *cur = root_;node *parent = cur;KeyOfT kot;while (cur){if (kot(data) > kot(cur->data_)){parent = cur;cur = cur->right_;}else if (kot(data) < kot(cur->data_)){parent = cur;cur = cur->left_;}else{return {cur, false};}}cur = new node(data);node *newnode = cur;cur->col_ = RED;if (kot(data) > kot(parent->data_)){parent->right_ = cur;}else{parent->left_ = cur;}cur->parent_ = parent;while (parent && parent->col_ == RED){node *grandfather = parent->parent_;if (grandfather->left_ == parent){node *uncle = grandfather->right_;// 叔叔存在且为红if (uncle && uncle->col_ == RED){// 变色+向上处理parent->col_ = uncle->col_ = BLACK;grandfather->col_ = RED;// 继续向上处理cur = grandfather;parent = cur->parent_;}else{// 叔叔不存在或者叔叔存在且为黑if (cur == parent->left_){// 右单旋rotater(grandfather);parent->col_ = BLACK;grandfather->col_ = RED;}else{// 左右双旋rotatelr(grandfather);cur->col_ = BLACK;grandfather->col_ = RED;}break;}}else if (grandfather->right_ == parent){node *uncle = grandfather->left_;// 叔叔存在且为红if (uncle && uncle->col_ == RED){// 变色+向上处理parent->col_ = uncle->col_ = BLACK;grandfather->col_ = RED;// 继续向上处理}else{// 叔叔不存在或者叔叔存在且为黑if (cur == parent->right_){// 左单旋rotatel(grandfather);parent->col_ = BLACK;grandfather->col_ = RED;}else{// 右左双旋rotaterl(grandfather);cur->col_ = BLACK;grandfather->col_ = RED;}break;}}else{assert(false);}}// 如果父亲不存在,说明cur为根,就要将cur置为黑色,如果父亲存在且为黑色则不需要处理;root_->col_ = BLACK;return {newnode, true};}template <class K, class V, class KeyOfT>void RBTree<K, V, KeyOfT>::_inorder(node *root){if (root == nullptr){return;}_inorder(root->left_);std::cout << root->data_;_inorder(root->right_);}template <class K, class V, class KeyOfT>bool RBTree<K, V, KeyOfT>::checkcolour(node *root, int blacknum, int reference){if (root == nullptr){if (blacknum != reference){return false;}return true;}if (root->col_ == BLACK){blacknum++;}if (root->col_ == RED && root->parent_->col_ == RED){KeyOfT kot;std::cout << kot(root->data_) << "出现连续红色节点" << std::endl;return false;}return checkcolour(root->left_, blacknum, reference) && checkcolour(root->right_, blacknum, reference);}template <class K, class V, class KeyOfT>bool RBTree<K, V, KeyOfT>::_isrbtree(node *root){if (root == nullptr){return true;}if (root->col_ != BLACK){return false;}int reference = 0;node *cur = root_;while (cur){if (cur->col_ == BLACK){reference++;}cur = cur->left_;}return checkcolour(root, 0, reference); // 检查连续的红色}
}namespace Map
{template <class K, class V>class map{struct MapKeyOfT{const K &operator()(const pair<const K, V> &kv){return kv.first;}};public:typedef typename RbTree::RBTree<K, pair<const K, V>, MapKeyOfT>::iterator iterator;typedef typename RbTree::RBTree<K, pair<const K, V>, MapKeyOfT>::const_iterator const_iterator;iterator begin(){return t_.begin();}iterator end(){return t_.end();}const_iterator begin() const{return t_.begin();}const_iterator end() const{return t_.end();}pair<iterator, bool> insert(const pair<const K, V> &kv){return t_.insert(kv);}V &operator[](const K &key){pair<iterator, bool> ret = insert({key, V()});return ret.first->second;}private:RbTree::RBTree<K, pair<const K, V>, MapKeyOfT> t_;};
}namespace Set
{template <class K>class set{struct SetKeyOfT{const K& operator()(const K& key){return key;}};public:typedef typename RbTree::RBTree<K, K, SetKeyOfT>::const_iterator iterator;typedef typename RbTree::RBTree<K, K, SetKeyOfT>::const_iterator const_iterator;iterator begin(){return t_.begin();}iterator end(){return t_.end();}const_iterator begin() const{return t_.begin();}const_iterator end() const{return t_.end();}pair<iterator, bool> insert(const K& key){pair<typename RbTree::RBTree<K, K, SetKeyOfT>::iterator, bool> ret = t_.insert(key);return ret;}private:RbTree::RBTree<K, K, SetKeyOfT> t_;};
}