C++之模拟实现map和set

简介: C++之模拟实现map和set

前言

基于之前的红黑树和map、set的相关知识,本节我们使用红黑树来模拟实现STL中的map和set。


一、迭代器

使用迭代器可以方便我们对数据结构进行遍历,它使数据结构的底层实与用户的使用分离(封装了底层实现),因此我们要给红黑树增加一个迭代器。

1.begin()和end()

STL中明确规定,begin()和end()代表的是一段前闭后开的区间。我们知道对红黑树进行中序遍历可以得到一个有序的序列,因此begin()可以放置在红黑树的最小节点处(即,最左节点),end()应该放置在红黑树最大节点的下一个位置。但是最大结点的下一个位置是什么呢?这个位置是nullptr吗?答案是不能是nullptr,因为对end()位置进行–操作要能找到最后一个元素,如果设置为nullptr就找不到最后一个结点了。

我们可以给红黑树增加一个header结点,让最大结点的next指向它

但是我们只是对它进行模拟,理解它的底层原理即可,为了不要让代码太过复杂,我们本次模拟实现就不设定header结点,直接让end()为nullptr即可(不实现–)。

2.operator++()

找迭代器的下一个结点(它的值一定比当前结点的值大)

template<class T, class Ref, class Ptr>
  struct __RBTreeIterator
  {
    typedef RBTreeNode<T> Node;
    typedef __RBTreeIterator<T, Ref, Ptr> Self;
    typedef __RBTreeIterator<T, T&, T*> iterator;
    Node* _node;
    __RBTreeIterator(Node* node)
      :_node(node)
    {}
    Ref operator*()
    {
      return _node->_data;
    }
    Ptr operator->()
    {
      return &_node->_data;
    }
    Self& operator++()
    {
      if (_node->_right)
      {
        Node* min = _node->_right;
        while (min->_left)
        {
          min = min->_left;
        }
        _node = min;
      }
      else
      {
        Node* cur = _node;
        Node* parent = _node->_parent;
        while (parent && cur == parent->_right)
        {
          cur = cur->_parent;
          parent = parent->_parent;
        }
        _node = parent;
      }
      return *this;
    }
    bool operator!=(const Self& s)
    {
      return _node != s._node;
    }
    bool operator==(const Self& s)
    {
      return _node == s._node;
    }
  };

二、改造红黑树

namespace Jinger
{
  enum Colour//结点颜色
  {
    RED,
    BLACK,
  };
  template<class K,class V>
  struct RBTreeNode//红黑树结点
  {
    pair<K, V> _kv;
    typedef RBTreeNode<K, V> Node;
    RBTreeNode(const pair<K,V>& kv)
      :_kv(kv)
      , _parent(nullptr)
      , _left(nullptr)
      , _right(nullptr)
      , _colour(RED)
    {}
    Node* _parent;
    Node* _left;
    Node* _right;
    Colour _colour;
  };
  template<class K, class V>
  struct __RBTreeIterator//迭代器
  {
    typedef RBTreeNode<K,V> Node;
    typedef __RBTreeIterator<K,V> Self;
    Node* _node;
    __RBTreeIterator(Node* node)
      :_node(node)
    {}
    pair<K,V>& operator*()
    {
      return _node->_kv;
    }
    pair<K, V>* operator->()
    {
      return &_node->_kv;
    }
    Self& operator++()
    {
      if (_node->_right)
      {
        _node = _node->_right;
      }
      else
      {
        Node* parent = _node->_parent;
        if (_node == parent->_left)
        {
          _node = parent;
        }
        else
        {
          while (parent && _node == parent->_right)
          {
            _node = parent;
            parent = parent->_parent;
          }
          _node = parent;
        }
      }
      return *this;
    }
    bool operator!=(const Self& s)
    {
      return _node != s._node;
    }
  };
  template<class K,class V>
  struct RBTree
  {
    typedef RBTreeNode<K,V> Node;
    typedef __RBTreeIterator<K,V> iterator;
    RBTree()
      :_root(nullptr)
    {}
    //左旋
    void RotateL(Node* parent)
    {
      Node* SubR = parent->_right;
      Node* SubRL = SubR->_left;
      parent->_right = SubRL;
      if (SubRL)
      {
        SubRL->_parent = parent;
      }
      Node* Grandpa = parent->_parent;
      SubR->_left = parent;
      parent->_parent = SubR;
      if (!Grandpa)
      {
        _root = SubR;
        SubR->_parent = nullptr;
      }
      else
      {
        if (parent == Grandpa->_left)
        {
          Grandpa->_left = SubR;
        }
        else
        {
          Grandpa->_right = SubR;
        }
      }
      SubR->_parent = Grandpa;
    }
    //右旋
    void RotateR(Node* parent)
    {
      Node* SubL = parent->_left;
      Node* SubLR = SubL->_right;
      Node* Grandpa = parent->_parent;
      parent->_parent = SubL;
      parent->_left = SubLR;
      if (SubLR)
      {
        SubLR->_parent = parent;
      }
      if (!Grandpa)
      {
        _root = SubL;
        SubL->_parent = nullptr;
      }
      else
      {
        if (parent == Grandpa->_left)
        {
          Grandpa->_left = SubL;
        }
        else
        {
          Grandpa->_right = SubL;
        }
      }
      SubL->_right = parent;
    }
    iterator begin()
    {
      Node* cur = _root;
      while (cur && cur->_left)
      {
        cur = cur->_left;
      }
      return iterator(cur);
    }
    iterator end()
    {
      return iterator(nullptr);
    }
    bool insert(const pair<K, V>& kv)
    {
      Node* newnode = new Node(kv);
      if (!_root)//空树
      {
        _root = newnode;
        _root->_colour = BLACK;
      }
      else
      {
        Node* parent = _root;
        Node* cur = _root;
        while (cur)
        {
          parent = cur;
          if (cur->_kv.first > kv.first)
          {
            cur = cur->_left;
          }
          else if (cur->_kv.first < kv.first)
          {
            cur = cur->_right;
          }
          else
          {
            return false;
          }
        }
        if (parent->_kv.first > kv.first)
        {
          parent->_left = newnode;
        }
        else
        {
          parent->_right = newnode;
        }
        newnode->_parent = parent;
        cur = newnode;
        parent = cur->_parent;
        if (parent->_colour == BLACK)//如果父亲的结点为黑色
        {
          return true;
        }
        while (parent && parent->_colour == RED)//如果parent为空,说明此时cur为根节点(如果调整到父节点为黑色就不需要再调整了)
        {
          Node* g = parent->_parent;//祖父
          Node* u = nullptr;//叔叔结点
          if (parent == g->_left)//如果父亲是祖父的左孩子,那么叔叔是祖父的右孩子
          {
            u = g->_right;
          }
          else
          {
            u = g->_left;
          }
          if (u && u->_colour == RED)
          {
            g->_colour = RED;
            u->_colour = parent->_colour = BLACK;
            cur = g;
            parent = cur->_parent;
          }
          else//叔叔不存在/叔叔的结点为黑色
          {
            //parent是g的右孩子,cur是parent的右孩子(左单旋)
            if (parent == g->_right && cur == parent->_right)
            {
              RotateL(g);
              parent->_colour = BLACK;
              g->_colour = RED;
            }
            //parent是g的左孩子,cur是parent的左孩子(右单旋)
            else if (parent == g->_left && cur == parent->_left)
            {
              RotateR(g);
              parent->_colour = BLACK;
              g->_colour = RED;
            }
            //parent是g的左孩子,cur是parent的右孩子(左右双旋)
            else if (parent == g->_left && cur == parent->_right)
            {
              RotateL(parent);
              RotateR(g);
              cur->_colour = BLACK;
              g->_colour = RED;
            }
            //parent是g的右孩子,cur是parent的左孩子(右左双旋)
            else if (parent == g->_right && cur == parent->_left)
            {
              RotateR(parent);
              RotateL(g);
              cur->_colour = BLACK;
              g->_colour = RED;
            }
            break;
          }
        }
      }
      _root->_colour = BLACK;//性质2要求根节点的颜色为黑色
      return true;
    }
    void inoder()
    {
      _inorder(_root);
    }
    void _inorder(Node* root)
    {
      if (!root) return;
      _inorder(root->_left);
      cout << root->_kv.first << ":" << root->_kv.second<< "  ";
      _inorder(root->_right);
    }
    bool _isbalance(Node* root, int count, const int& leftcount)
    {
      if (root == nullptr)
      {
        if (count != leftcount)
        {
          cout << "每条路径黑色结点数不相同,违反了性质4" << endl;
          return false;
        }
        else
        {
          return true;
        }
      }
      if (root->_colour == RED && root->_parent->_colour == RED)
      {
        cout << "父亲结点和cur都是红色,违反了性质3" << endl;
        return false;
      }
      if (root->_colour == BLACK)
      {
        count++;
      }
      bool left = _isbalance(root->_left, count, leftcount);
      bool right = _isbalance(root->_right, count, leftcount);
      return left && right;
    }
    bool isBalance()
    {
      if (_root == nullptr)
      {
        return true;
      }
      if (_root->_colour == RED)
      {
        cout << "根节点为红色,违反了性质2" << endl;
        return false;
      }
      int leftcount = 0;
      Node* cur = _root;
      while (cur->_left)
      {
        if (cur->_colour == BLACK)
        {
          leftcount++;
        }
        cur = cur->_left;
      }
      cout << leftcount << endl;
     return _isbalance(_root, 0, leftcount);
    }
  private:
    Node* _root;
  };
}

三、map的模拟实现

map的底层结构就是一个红黑树,因此在map中直接封装一个红黑树,然后包装一下它的借口即可。

namespace Jinger
{
  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<K, pair< const K, V>, MapKeyOfT>::iterator iterator;
    iterator begin()
    {
      return _t.begin();
    }
    iterator end()
    {
      return _t.end();
    }
    pair<iterator, bool> insert(const pair<const K, V>& kv)
    {
      return _t.insert(kv);
    }
    V& operator[](const K& k)
    {
      pair<iterator, bool> ret = _t.insert(make_pair(k, V()));
      return ret.first->second;
    }
    iterator& find(const K& k)
    {
      _t.find(k);
    }
  private:
    Jinger::RBTree<K, pair<const K, V>, MapKeyOfT> _t;
  };
}

四、set的模拟实现

set的底层结构就是一个红黑树,因此在map中直接封装一个红黑树,然后包装一下它的借口即可。

namespace Jinger
{
  template<class K>
  class set
  {
    struct SetKeyOfT
    {
      const K& operator()(const K& key)
      {
        return key;
      }
    };
  public:
    typedef typename RBTree<K, K, SetKeyOfT>::const_iterator iterator;
    typedef typename RBTree<K, K, SetKeyOfT>::const_iterator const_iterator;
    iterator begin() const
    {
      return _t.begin();
    }
    iterator end() const
    {
      return _t.end();
    }
    pair<iterator, bool> insert(const K& key)
    {
      pair<typename RBTree<K,K,SetKeyOfT>::iterator,bool>  ret = _t.insert(key);
      return pair<iterator, bool>(ret.first, ret.second);
    }
  private:
    RBTree<K, K, SetKeyOfT> _t;
  };
}

总结

以上就是今天要讲的内容,本文介绍了如何用红黑树模拟实现map和set的相关概念。本文作者目前也是正在学习C++相关的知识,如果文章中的内容有错误或者不严谨的部分,欢迎大家在评论区指出,也欢迎大家在评论区提问、交流。

最后,如果本篇文章对你有所启发的话,希望可以多多支持作者,谢谢大家!

相关文章
|
1月前
|
算法
你对Collection中Set、List、Map理解?
你对Collection中Set、List、Map理解?
65 18
你对Collection中Set、List、Map理解?
|
1月前
|
存储 缓存 安全
只会“有序无序”?面试官嫌弃的List、Set、Map回答!
小米,一位热衷于技术分享的程序员,通过与朋友小林的对话,详细解析了Java面试中常见的List、Set、Map三者之间的区别,不仅涵盖了它们的基本特性,还深入探讨了各自的实现原理及应用场景,帮助面试者更好地准备相关问题。
60 20
|
2月前
|
存储 C++ 容器
【C++】map、set基本用法
本文介绍了C++ STL中的`map`和`set`两种关联容器。`map`用于存储键值对,每个键唯一;而`set`存储唯一元素,不包含值。两者均基于红黑树实现,支持高效的查找、插入和删除操作。文中详细列举了它们的构造方法、迭代器、容量检查、元素修改等常用接口,并简要对比了`map`与`set`的主要差异。此外,还介绍了允许重复元素的`multiset`和`multimap`。
37 3
【C++】map、set基本用法
|
2月前
|
存储 算法 C++
【C++】unordered_map(set)
C++中的`unordered`容器(如`std::unordered_set`、`std::unordered_map`)基于哈希表实现,提供高效的查找、插入和删除操作。哈希表通过哈希函数将元素映射到特定的“桶”中,每个桶可存储一个或多个元素,以处理哈希冲突。主要组成部分包括哈希表、哈希函数、冲突处理机制、负载因子和再散列,以及迭代器。哈希函数用于计算元素的哈希值,冲突通过开链法解决,负载因子控制哈希表的扩展。迭代器支持遍历容器中的元素。`unordered_map`和`unordered_set`的插入、查找和删除操作在理想情况下时间复杂度为O(1),但在冲突较多时可能退化为O(n)。
29 5
|
2月前
|
存储 C++ 容器
【C++】map的模拟实现
C++中的`map`是STL中的一种关联容器,存储键值对且键唯一。`map`基于红黑树实现,自动按键排序,支持动态调整、复杂数据类型、丰富的成员函数及双向迭代器。插入、查找等操作保证了对数时间复杂度,适用于需要快速查找和有序存储的场景。
26 3
|
2月前
|
存储 C++ 容器
【C++】set模拟实现
C++中的`set`是STL提供的一种关联容器,用于存储唯一元素并自动按特定顺序(默认升序)排序。其内部通过红黑树实现,保证了高效的插入、删除和查找操作,时间复杂度均为O(log n)。`set`支持迭代器遍历,提供了良好的数据访问接口。
43 3
|
4月前
|
存储 Java API
【数据结构】map&set详解
本文详细介绍了Java集合框架中的Set系列和Map系列集合。Set系列包括HashSet(哈希表实现,无序且元素唯一)、LinkedHashSet(保持插入顺序的HashSet)、TreeSet(红黑树实现,自动排序)。Map系列为双列集合,键值一一对应,键不可重复,值可重复。文章还介绍了HashMap、LinkedHashMap、TreeMap的具体实现与应用场景,并提供了面试题示例,如随机链表复制、宝石与石头、前K个高频单词等问题的解决方案。
56 6
【数据结构】map&set详解
|
3月前
|
存储 JavaScript 前端开发
Set、Map、WeakSet 和 WeakMap 的区别
在 JavaScript 中,Set 和 Map 用于存储唯一值和键值对,支持多种操作方法,如添加、删除和检查元素。WeakSet 和 WeakMap 则存储弱引用的对象,有助于防止内存泄漏,适合特定场景使用。
|
3月前
|
存储 缓存 Java
【用Java学习数据结构系列】HashMap与TreeMap的区别,以及Map与Set的关系
【用Java学习数据结构系列】HashMap与TreeMap的区别,以及Map与Set的关系
49 1
|
4月前
|
算法
你对Collection中Set、List、Map理解?
你对Collection中Set、List、Map理解?
45 5