【C++】map和set封装

简介: 【C++】map和set封装

> 作者:დ旧言~

> 座右铭:松树千年终是朽,槿花一日自为荣。

> 目标:手撕哈希表的闭散列和开散列

> 毒鸡汤:学习,学习,再学习 ! 学,然后知不足。

> 专栏选自:C嘎嘎进阶

> 望小伙伴们点赞👍收藏✨加关注哟💕💕



🌟前言  

map和set的知识我们学习了,模拟实现我们已经实现了AVL树和红黑树。熟练知识点为map和set的使用提供了前提,手撕AVL树和红黑树让我了解到map和set的底层如何实现,有了知识点和两树的模拟铺垫,那我们该如何封装map和set呢?


⭐主体

学习map和set封装咱们按照下面的图解:



🌙map/set底层原理

Map和Set底层是用红黑树封装的,而红黑树结构是KV结构。

RBTree是通过传入的Value的值来判断类型,也就是一棵泛型的RBTree,通过不同的实例化,实现出了Map和Set,对于map:传key,对于set:传pair,如图:



简化后的map源码:



简化后的set源码:



我们查看源码后,因此在封装我们map和set时,让我们的红黑树增加一个模板参数T来识别map和set,代码如下:(此处的代码是修改了红黑树)

template<class K, class T>
class RBTree


分析原理:

如果是set容器,那么它传入底层红黑树的模板参数就是Key和Key:

template<class K>
class set
{
  private:
    RBTree<K,K> _t;
};


如果是map容器,传入底层红黑树的模板参数就是Key和Key和value的键值对:

class map
{
private:
    RBTree<K, pair<const K,V>> _t;
};


问题拓展:

通过上面,我们可以知道,对于set和map的区别:我们只要通过第二个模板参数就能进行区分那是不是第一个模板参数就没有意义了呢?

  • 对于insert(const Value&v)来说,需要放入存入的值,确实是这个样子的,插入的值是value,对于set就是key,对于map就是pair。
  • 但是对于find(const Key&key)来说,查找的参数不是value,找的不是pair而是Key,对于map容器来说就不行了。


🌙map/set定义



map和set封装都用头文件来,代码如下:

set:

template<class K>
class set
{
  private:
    RBTree<K,K> _t;
};


map:

class map
{
private:
    RBTree<K, pair<const K,V>> _t;
};


🌙map/set仿函数

问题阐述:

插入的时候data的大小如何去进行比较:我们并不知道是什么类型是key,还是pair的比较,而我们刚开始kv结构就直接用kv.first去比较了。

  • 对于set是Key,可以比较
  • 对于map是pair,那我们要取其中的first来比较,但是pair的大小并不是直接按照first去进行比较的,而我们只需要按照first去进行比较


问题分析:

由于底层的红黑树不知道传的是map还是set容器,当需要进行两个结点键值的比较时,底层红黑树传入的仿函数来获取键值Key,进行两个结点键值的比较:这个时候我们就需要仿函数了,如果是set那就是用于返回T当中的键值Key,如果是map那就是用于返回pair的first:


注意事项:

仿函数/函数对象也是类,是一个类对象。仿函数要重载operator()。


代码实现:

map:

namespace lyk
{
  template<class K, class V>
  class map
  {
    struct MapkeyOfT // 仿函数
    {
      const K& operator()(const pair<const K, V>& kv)
      {
        return kv.first;
      }
    };
}


set:

namespace lyk
{
  template <class K>
  class set  // 仿函数,重载operator()
  {
    struct SetKeyOfT
    {
      const K& operator()(const K& key)
      {
        return key;
      }
 
    };
}


图示:



我们有了仿函数时,查找函数就可以用仿函数来替换比较部分

KeyOfT kot;//仿函数对象
Node* parent = nullptr;
Node* cur = _root;
while (cur)
{
  if (kot(cur->_data)<kot(data))
  {
    parent = cur;
    cur = cur->_right;
  }
  else if (kot(cur->_data)>kot(data))
  {
    parent = cur;
    cur = cur->_left;
  }
  else
  {
    return false;
  }
}


🌙map/set插入

接下来 map/set 的插入直接套用红黑树的即可,代码如下:

//map的插入,插入pair
bool insert(const pair<K, V>& kv)
{
  return _t.Insert(kv);
}
 
//set的插入,插入key
bool insert(const K& key)
{
  return _t.Insert(key);
}


🌙map/set迭代器

💫迭代器的定义

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)
  {}
 
  //普通迭代器的时候,它是拷贝构造
  //const迭代器的时候,它是构造,支持用普通迭代器构造const迭代器
  __RBTreeIterator(const iterator& s)
    :_node(s._node)
  {}
}


💫解引用操作

作用:返回对应结点数据的引用

Ref operator*()
  {
    return _node->_data;
  }


💫成员访问操作符

作用:返回结点数据的引用

Ptr operator->()
  {
    return &_node->_data;
  }


💫!=、==

  bool operator !=(const Self & s) const
  {
    return _node != s._node;
  }
  bool operator ==(const Self& s) const
  {
    return _node == s._node;
  }


💫begin() 与 end()

begin():返回中序(左、根、右)第一个结点的正向迭代器,即最左节点,返回的是最左节点,直接找最左节点即可

end():返回中序(左、根、右)最后一个结点下一个位置的正向迭代器,这里直接用空指针

template<class K, class T,class KeyOfT>
class RBTree
{
  typedef RBTreeNode<T> Node;
public:
  typedef __RBTreeIterator<T> iterator;
 
  iterator begin()
  {
    Node* left = _root;
    while (left && left->_left)
    {
      left = left->_left;
    }
    return iterator(left);
  }
 
  iterator end()
  {
    return iterator(nullptr);
  }
}


💫迭代器的++

一个结点的正向迭代器进行++操作后,根据红黑树中序(左、根、右)找到当前结点的下一个结点,中序的第一个节点是最左,迭代器的++怎么去找:

  • 如果节点的右子树不为空++就是找右子树的最左节点
  • 如果节点的右子树为空++就是找祖先(孩子是父亲的左的那个祖先)
  Self& operator++()
  {
    if (_node->_right)
    {
      Node* min = _node->_right;
      while (min->_left)
      {
        min = min->_left;
      }
      _node = min;
    }
    else
    {
      Node* cur = _node;
      Node* parent = cur->_parent;
      while (parent && cur == parent->_right)
      {
        cur = cur->_parent;
        parent = parent->_parent;
      }
      _node = parent;
    }
    return *this;
  }


💫迭代器的--

对于–,如果是根,–就是左子树,找到左子树最大的那一个(最右节点)

  • 如果节点的左子树不为空,--找左子树最右的节点
  • 如果节点的左子树为空,--找祖先(孩子是父亲的右的祖先)
Self& operator--()
  {
    if (_node->_left)
    {
      Node* max = _node->_left;
      while (max->_right)
      {
        max = max->_right;
      }
      _node = max;
    }
    else
    {
      Node* cur = _node;
      Node* parent = cur->_parent;
      while (parent&&cur==parent->_left)
      {
        cur = cur->_parent;
        parent = parent->_parent;
      }
      _node = parent;
    }
    return *this;
  }


🌙源代码及其测试

💫map代码

#include"RBTree.h"
 
namespace lyk
{
  template<class K, class V>
  class map
  {
    struct MapkeyOfT // 仿函数
    {
      const K& operator()(const pair<const K, V>& kv)
      {
        return kv.first;
      }
    };
  public:
    //typename:没有实例化的模板,区分不了是静态变量还是类型,typename告诉编译器是类型
    typedef typename RBTree<K, pair<const K, V>, MapkeyOfT>::iterator iterator;
    typedef typename 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(make_pair(key, V()));
      return ret.first->second;
    }
  private:
    RBTree<K, pair<const K, V>, MapkeyOfT> _t;
  };
 
  void test_map1() // 测试代码
  {
    map<int, int> m;
    int a[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };
    for (auto e : a)
    {
      m.insert(make_pair(e, e));
    }
 
    map<int, int>::iterator it = m.begin();
    while (it != m.end())
    {
      //it->first += 100;
      it->second += 100;
 
      cout << it->first << ":" << it->second << endl;
      ++it;
    }
    cout << endl;
  }
}


💫set代码

#include"RBTree.h"
 
namespace lyk
{
  template <class K>
  class set  // 仿函数,重载operator()
  {
    struct SetKeyOfT
    {
      const K& operator()(const K& key)
      {
        return key;
      }
 
    };
  
  public:
    //typename:没有实例化的模板,区分不了是静态变量还是类型,typename告诉编译器是类型
    typedef typename RBTree<K, K, SetKeyOfT>::const_iterator iterator;//key不可以修改
    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)
    {
      //底层红黑树的iterator是普通迭代器
      pair<typename RBTree<K, K, SetKeyOfT>::iterator, bool> ret = _t.Insert(key);
      return pair<iterator, bool>(ret.first, ret.second);//用普通迭代器构造const迭代器
    }
 
  private:
    RBTree<K, K, SetKeyOfT> _t;
  
  };
 
  void test_set1() // 测试代码
  {
    set<int> s;
    int a[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };
    for (auto e : a)
    {
      s.insert(e);
    }
 
    set<int>::iterator it = s.begin();
    while (it != s.end())
    {
      cout << *it << " ";
      ++it;
    }
    cout << endl;
  }
}


💫红黑树代码

#pragma once
 
#include <iostream>
#include <assert.h>
#include <time.h>
using namespace std;
enum Color
{
  RED,
  BLACK,
};
template<class T>
struct RBTreeNode
{
  T _data;
  RBTreeNode<T>* _left;
  RBTreeNode<T>* _right;
  RBTreeNode<T>* _parent;
 
  Color _col;
  RBTreeNode(const T& data)
    :_data(data)
    , _left(nullptr)
    , _right(nullptr)
    , _parent(nullptr)
    , _col(RED)
  {}
};
 
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)
  {}
 
 
  //普通迭代器的时候,它是拷贝构造
  //const迭代器的时候,它是构造,支持用普通迭代器构造const迭代器
  __RBTreeIterator(const iterator& s)
    :_node(s._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 = cur->_parent;
      while (parent && cur == parent->_right)
      {
        cur = cur->_parent;
        parent = parent->_parent;
      }
      _node = parent;
    }
    return *this;
  }
 
  Self& operator--()
  {
    if (_node->_left)
    {
      Node* max = _node->_left;
      while (max->_right)
      {
        max = max->_right;
      }
      _node = max;
    }
    else
    {
      Node* cur = _node;
      Node* parent = cur->_parent;
      while (parent && cur == parent->_left)
      {
        cur = cur->_parent;
        parent = parent->_parent;
      }
      _node = parent;
    }
    return *this;
  }
 
  bool operator !=(const Self& s) const
  {
    return _node != s._node;
  }
  bool operator ==(const Self& s) const
  {
    return _node == s._node;
  }
};
 
 
template<class K, class T, class KeyOfT>
class RBTree
{
  typedef RBTreeNode<T> Node;
public:
  typedef __RBTreeIterator<T, T&, T*> iterator;
  typedef __RBTreeIterator<T, const T&, const T*> const_iterator;
  const_iterator begin() const
  {
    Node* left = _root;
    while (left && left->_left)
    {
      left = left->_left;
    }
    return const_iterator(left);
 
  }
  const_iterator end() const
  {
    return const_iterator(nullptr);
  }
 
  iterator begin()
  {
    Node* left = _root;
    while (left && left->_left)
    {
      left = left->_left;
    }
    return iterator(left);
 
  }
  iterator end()
  {
    return iterator(nullptr);
  }
  pair<iterator, bool> Insert(const T& data)
  {
    if (_root == nullptr)
    {
      _root = new Node(data);
      _root->_col = BLACK;
      return make_pair(iterator(_root), true);
    }
    KeyOfT kot;
    Node* parent = nullptr;
    Node* cur = _root;
    while (cur)
    {
      if (kot(cur->_data) < kot(data))
      {
        parent = cur;
        cur = cur->_right;
      }
      else if (kot(cur->_data) > kot(data))
      {
        parent = cur;
        cur = cur->_left;
      }
      else
      {
        return make_pair(iterator(cur), false);
      }
    }
    cur = new Node(data);
    Node* newnode = cur;
    cur->_col = RED;
    if (kot(parent->_data) < kot(data))
    {
      parent->_right = cur;
      cur->_parent = parent;
    }
    else
    {
      parent->_left = cur;
      cur->_parent = parent;
    }
 
    while (parent && parent->_col == RED)
    {
      Node* grandfater = parent->_parent;
      if (parent == grandfater->_left)
      {
        Node* uncle = grandfater->_right;
        //情况一:u存在且为红
        if (uncle && uncle->_col == RED)
        {
          parent->_col = uncle->_col = BLACK;
          grandfater->_col = RED;
          //向上调整
          cur = grandfater;
          parent = cur->_parent;
        }
        else
        {
          //情况2
          if (cur == parent->_left)
          {
            RotateR(grandfater);
            parent->_col = BLACK;
            grandfater->_col = RED;
          }
          //情况3
          else
          {
            //       g
            //  p
            //    c 
            RotateL(parent);
            RotateR(grandfater);
            cur->_col = BLACK;
            grandfater->_col = RED;
          }
          break;
        }
      }
      else//parent==grandfater->_right
      {
        Node* uncle = grandfater->_left;
        //情况1:u存在且为红色
        if (uncle && uncle->_col == RED)
        {
          uncle->_col = parent->_col = BLACK;
          grandfater->_col = RED;
          //向上调整
          cur = grandfater;
          parent = cur->_parent;
        }
        else
        {
          //情况2:u不存在/u存在为黑色
          //g
          //    p
          //        c
          if (cur == parent->_right)
          {
            RotateL(grandfater);
            grandfater->_col = RED;
            parent->_col = BLACK;
          }
          //情况3
          //     g
           //         p
           //      c
          else
          {
            RotateR(parent);
            RotateL(grandfater);
            cur->_col = BLACK;
            grandfater->_col = RED;
          }
          break;
        }
 
      }
    }
    //根变黑
    _root->_col = BLACK;
    return make_pair(iterator(newnode), true);
  }
 
  void RotateL(Node* parent)
  {
    Node* subR = parent->_right;
    Node* subRL = subR->_left;
    parent->_right = subRL;
    if (subRL)
      subRL->_parent = parent;
    Node* ppNode = parent->_parent;
    subR->_left = parent;
    parent->_parent = subR;
    if (ppNode == nullptr)
    {
      _root = subR;
      _root->_parent = nullptr;
    }
    else
    {
      if (ppNode->_left == parent)
      {
        ppNode->_left = subR;
      }
      else
      {
        ppNode->_right = subR;
      }
      subR->_parent = ppNode;
    }
  }
 
  void RotateR(Node* parent)
  {
    Node* subL = parent->_left;
    Node* subLR = subL->_right;
    parent->_left = subLR;
    if (subLR)
      subLR->_parent = parent;
    Node* ppNode = parent->_parent;
    parent->_parent = subL;
    subL->_right = parent;
    if (ppNode == nullptr)
    {
      _root = subL;
      _root->_parent = nullptr;
    }
    else
    {
      if (ppNode->_left == parent)
      {
        ppNode->_left = subL;
      }
      else
      {
        ppNode->_right = subL;
      }
      subL->_parent = ppNode;
    }
  }
 
 
  void InOrder()
  {
    _InOrder(_root);
  }
  void _InOrder(Node* root)
  {
    if (root == nullptr)
      return;
    _InOrder(root->_left);
    cout << root->_kv.first << ":" << root->_kv.second << endl;
    _InOrder(root->_right);
  }
 
 
  bool Check(Node* root, int blackNum, int ref)
  {
    if (root == nullptr)
    {
      //cout << blackNum << endl;
      if (blackNum != ref)
      {
        cout << "违反规则:本条路径的黑色结点的数量根最左路径不相等" << endl;
        return false;
      }
      return true;
    }
    if (root->_col == RED && root->_parent->_col == RED)
    {
      cout << "违反规则:出现连续的红色结点" << endl;
      return false;
    }
    if (root->_col == BLACK)
    {
      ++blackNum;
    }
    return Check(root->_left, blackNum, ref)
      && Check(root->_right, blackNum, ref);
  }
 
  bool IsBalance()
  {
    if (_root == nullptr)
    {
      return true;
    }
 
    if (_root->_col != BLACK)
    {
      return false;
    }
    int ref = 0;
    Node* left = _root;
    while (left)
    {
      if (left->_col == BLACK)
      {
        ++ref;
      }
      left = left->_left;
    }
    return Check(_root, 0, ref);
  }
private:
  Node* _root = nullptr;
};
 


💫测试

测试set:



测试map:



🌟结束语

      今天内容就到这里啦,时间过得很快,大家沉下心来好好学习,会有一定的收获的,大家多多坚持,嘻嘻,成功路上注定孤独,因为坚持的人不多。那请大家举起自己的小手给博主一键三连,有你们的支持是我最大的动力💞💞💞,回见。


目录
相关文章
|
3月前
|
编译器 C++ 容器
【c++丨STL】基于红黑树模拟实现set和map(附源码)
本文基于红黑树的实现,模拟了STL中的`set`和`map`容器。通过封装同一棵红黑树并进行适配修改,实现了两种容器的功能。主要步骤包括:1) 修改红黑树节点结构以支持不同数据类型;2) 使用仿函数适配键值比较逻辑;3) 实现双向迭代器支持遍历操作;4) 封装`insert`、`find`等接口,并为`map`实现`operator[]`。最终,通过测试代码验证了功能的正确性。此实现减少了代码冗余,展示了模板与仿函数的强大灵活性。
99 2
|
3月前
|
存储 算法 C++
【c++丨STL】map/multimap的使用
本文详细介绍了STL关联式容器中的`map`和`multimap`的使用方法。`map`基于红黑树实现,内部元素按键自动升序排列,存储键值对,支持通过键访问或修改值;而`multimap`允许存在重复键。文章从构造函数、迭代器、容量接口、元素访问接口、增删操作到其他操作接口全面解析了`map`的功能,并通过实例演示了如何用`map`统计字符串数组中各元素的出现次数。最后对比了`map`与`set`的区别,强调了`map`在处理键值关系时的优势。
201 73
|
13天前
|
存储 JavaScript 前端开发
for...of循环在遍历Set和Map时的注意事项有哪些?
for...of循环在遍历Set和Map时的注意事项有哪些?
34 0
|
13天前
|
存储 C++ 容器
unordered_set、unordered_multiset、unordered_map、unordered_multimap的介绍及使用
unordered_set是不按特定顺序存储键值的关联式容器,其允许通过键值快速的索引到对应的元素。在unordered_set中,元素的值同时也是唯一地标识它的key。在内部,unordered_set中的元素没有按照任何特定的顺序排序,为了能在常数范围内找到指定的key,unordered_set将相同哈希值的键值放在相同的桶中。unordered_set容器通过key访问单个元素要比set快,但它通常在遍历元素子集的范围迭代方面效率较低。它的迭代器至少是前向迭代器。前向迭代器的特性。
|
13天前
|
编译器 C++ 容器
用一棵红黑树同时封装出map和set
再完成上面的代码后,我们的底层代码已经完成了,这时候已经是一个底层STL的红黑树了,已经已符合库里面的要求了,这时候我们是需要给他穿上对应的“衣服”,比如穿上set的“衣服”,那么这个穿上set的“衣服”,那么他就符合库里面set的要求了,同样map一样,这时候我们就需要实现set与map了。因此,上层容器map需要向底层红黑树提供一个仿函数,用于获取T当中的键值Key,这样一来,当底层红黑树当中需要比较两个结点的键值时,就可以通过这个仿函数来获取T当中的键值了。我们就可以使用仿函数了。
|
13天前
|
存储 编译器 容器
set、map、multiset、multimap的介绍及使用以及区别,注意事项
set是按照一定次序存储元素的容器,使用set的迭代器遍历set中的元素,可以得到有序序列。set当中存储元素的value都是唯一的,不可以重复,因此可以使用set进行去重。set默认是升序的,但是其内部默认不是按照大于比较,而是按照小于比较。set中的元素不能被修改,因为set在底层是用二叉搜索树来实现的,若是对二叉搜索树当中某个结点的值进行了修改,那么这棵树将不再是二叉搜索树。
|
3月前
|
存储 算法 C++
【c++丨STL】set/multiset的使用
本文深入解析了STL中的`set`和`multiset`容器,二者均为关联式容器,底层基于红黑树实现。`set`支持唯一性元素存储并自动排序,适用于高效查找场景;`multiset`允许重复元素。两者均具备O(logN)的插入、删除与查找复杂度。文章详细介绍了构造函数、迭代器、容量接口、增删操作(如`insert`、`erase`)、查找统计(如`find`、`count`)及`multiset`特有的区间操作(如`lower_bound`、`upper_bound`、`equal_range`)。最后预告了`map`容器的学习,其作为键值对存储的关联式容器,同样基于红黑树,具有高效操作特性。
137 3
|
4月前
|
编译器 容器
哈希表模拟封装unordered_map和unordered_set
哈希表模拟封装unordered_map和unordered_set
|
4月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
13天前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;

热门文章

最新文章

下一篇
oss创建bucket
OSZAR »