map和set 的封装
创始人
2024-05-28 11:50:42
0

文章目录

  • 引入
    • key-value模型
    • map和set底层
  • set
    • set的几个重要接口
  • map
    • map几个重要的接口
  • map和set的封装

引入

对于map和set的引入,我们用一道在程序中常见的问题解决:
给定一个数组int arr[]={1,2,1,3,1,4,1,5,5,2,3,4,5};,给出以下问题的解决方案:

  1. 找出出现次数最多的元素
  2. 去除数组中重复的元素,并输出(任意顺序)

这些问题在没有学习map和set之前并没有很好的解决方法(但是map和set并不是唯一的解决方法),第二个问题:之前我们解决的方法一般就开辟一个新数组tmp,遍历arr中所有元素,对于arr中任意元素遍历tmp数组如果元素在tmp中未出现就插入。第一个问题,只需将数组类型改成一个结构体,该结构体包含一个元素和他出现的次数,则在最后一步如果arr中元素已经出现在tmp中,就改变tmp中该元素对应结构体的次数。

我们发现arr在tmp中插入时每一次都要遍历一遍tmp数组,是否能提升查找效率呢?

所以我们现在需要一个容器:

  • 该容器中不存在重复元素
  • 该容器中查找一个元素效率要高

key-value模型

key-value模型适用于上面的问题1,每一个元素(key)和他出现的次数(value)一 一对应,但是每次查找tmp数组,都是以key为依据查找。
举个例子:一个英汉字典就是典型的key-value模型,英文(key)是你搜索的依据,中文(value)是查找的内容,两者一一对应

map和set底层

有一个数据结构完美符合上面我们需要容器的两个条件——红黑树。首先红黑树不能插入重复元素,所以可以做到去重的效果。又因为满足二叉搜索树所以查找效率比高。而且比二叉搜索树稳定。所以map和set的底层采用的是红黑树

set

set存储元素的类型为key,容器中将该类型定义为value_type

set的几个重要接口


  • insert
     pair insert (const value_type& val);pair insert (value_type&& val);
    
    注意insert的返回值是一个pair类型,first是插入元素的迭代器(如果val在set中不存在返回插入新元素的迭代器,如果在set中存在返回set中值为val的那个元素的迭代器),second代表是否插入成功

  • operator++
    由于set底层采用的是红黑树,迭代器++则采用的是树的中序遍历,所以遍历的结果应该是一个有序数组

map

map中存储的元素的类型是:pair,容器中将该类型定义为value_type

map几个重要的接口


  • insert
    pair insert (const value_type& val);
    template  
    pair insert (P&& val);
    
    返回值
    注意insert的返回值是一个pair类型,first是插入元素的迭代器(如果val在set中不存在返回插入新元素的迭代器,如果在set中存在返回set中值为val的那个元素的迭代器),second代表是否插入成功

  • find
    iterator find (const key_type& k);
    const_iterator find (const key_type& k) const;
    
    返回值
    如果找到了返回该元素的迭代器,找不到返回map::end()

  • operator[]
    mapped_type& operator[] (const key_type& k);
    mapped_type& operator[] (key_type&& k);
    
    这里mapped_type是我们map中存储元素类型pair中的value
    这个运算符重载给我们map的使用带来了巨大的方便,map[key](map.insert(key,value()).first)->second是等价的。

  • operator++
    由于map底层采用的是红黑树,迭代器++则采用的是树的中序遍历,所以遍历的结果应该是一个有序数组

map和set的封装

我们对于map和set封装始终是在红黑树的基础上进行封装的,红黑树的代码可以参考blog红黑树
红黑树的代码接下来的封装都是在这个代码的基础上进行修改的!
封装的结构为:
在这里插入图片描述

这里以map的封装为例: map的封装弄明白了,set就非常简单了
首先需要将红黑树的节点模板改一下:
在这里插入图片描述
这里修改后使得树节点里面存储的是ValueType类型,ValueType类型实际上就是修改前pair,我们可以发现ValueType实际上就是我们在外层插入时插入时传进去的类型

template struct TreeNode{};template     
class RBTree{};

那么问题来了:

  • 已经传入了ValueType为什么要传入K?
    传入Key类型只是作为返回值类型,因为find函数参数为const K& x,仅此而已。

  • 红黑树在插入搜索时需要比较key值,节点里面存储的是ValueType,如何比较?
    有些同学会认为很简单,直接(ValueType)x.second不就行了?这样确实可以,但是没有考虑一个问题set也需要使用这份红黑树代码作为底层,如果用上面的代码,set中的ValueTypeK是同一个类型,这时就会出现错误。
    我们的解决方法是使用仿函数Get_key来取出ValueType中的Value,让上层来决定如何取出方法。这就是仿函数的妙处

剩下来只需要将红黑树中函数封装到上层map中就可以了


#include "RedBlackTree.h"
#include namespace sht
{template class map{struct Get_Key_From_ValueType{Key operator()(const std::pair &v){return v.first;}};public:typedef std::pair ValueType;    //注意这里定义的ValueType是实际树节点里面存的内容typedef typename sht::RBTree::iterator iterator;typedef typename sht::RBTree::const_iterator const_iterator;std::pair insert(const std::pair x){return tree.insert(x);}Value&  operator [](const Key& x){auto ret = tree.insert(std::make_pair(x, Value()));return (ret.first)->second;}const_iterator begin() const // 常量迭代器{return tree.begin();}const_iterator end() const{return tree.end();}iterator begin() // 普通迭代器 : 注意普通迭代器的key也是无法修改的{return tree.begin();}iterator end(){return tree.end();}private:sht::RBTree tree;};
}

这里map还有一个设计的技巧如果是const对象,那么beginend要返回const迭代器,这里重载了begin和end函数(参考这篇blog)。这个设计很巧妙😋

接下来是一个硬骨头:迭代器的设计

   template    //ref是class RBTreeiterator{public:typedef TreeNode Node;typedef RBTreeiterator Self;RBTreeiterator(Node *x=nullptr){p = x;}//迭代器的运算符重载Self &operator++();Self &operator--();bool operator==(const RBTreeiterator &x)bool operator!=(const RBTreeiterator &x)T operator*()ptr operator->()     Node *p;};

然而一个问题出现了,非const对象想要使用const迭代器时就会出现问题:
例如:map::const_iterator it=x.begin();这时就会报错,因为是非const对象,调用的begin是非const成员函数,返回的是普通迭代器,所以报错的原因是缺乏普通迭代器到const迭代器的转化!!
在这里插入图片描述
重写一个拷贝构造函数就完美解决了,在迭代器里面的iterator巧妙的构造出了普通迭代器类型,很巧妙的解决了该问题

迭代器的运算符重载中还有一个难点是operator ++——即按中序取当前节点的下一个节点,这种问题参考LeetCode剑指 Offer II 055. 二叉搜索树迭代器,我这里就不赘述了。

相关内容

热门资讯

【NI Multisim 14...   目录 序言 一、工具栏 🍊1.“标准”工具栏 🍊 2.视图工具...
银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
AWSECS:访问外部网络时出... 如果您在AWS ECS中部署了应用程序,并且该应用程序需要访问外部网络,但是无法正常访问,可能是因为...
Android|无法访问或保存... 这个问题可能是由于权限设置不正确导致的。您需要在应用程序清单文件中添加以下代码来请求适当的权限:此外...
北信源内网安全管理卸载 北信源内网安全管理是一款网络安全管理软件,主要用于保护内网安全。在日常使用过程中,卸载该软件是一种常...
AWSElasticBeans... 在Dockerfile中手动配置nginx反向代理。例如,在Dockerfile中添加以下代码:FR...
AsusVivobook无法开... 首先,我们可以尝试重置BIOS(Basic Input/Output System)来解决这个问题。...
ASM贪吃蛇游戏-解决错误的问... 要解决ASM贪吃蛇游戏中的错误问题,你可以按照以下步骤进行:首先,确定错误的具体表现和问题所在。在贪...
月入8000+的steam搬砖... 大家好,我是阿阳 今天要给大家介绍的是 steam 游戏搬砖项目,目前...