【C++】STL map和set用法基本介绍

news2025/1/11 12:59:02

map、set用法简介

  • 前言
  • 正式开始
    • set
      • 构造
      • erase
      • find
      • swap
      • count
      • lower_bound 和 upper_bound
        • lower_bound
        • upper_bound
      • equal_range
        • pair
      • multiuset
        • find
        • erase
        • count
    • map
      • 构造
      • insert
      • [ ]
        • [ ]底层原理
    • multimap
    • 两道题目
      • 前K个高频单词
      • 两个数组的交集

在这里插入图片描述

前言

首先,使用map和set最少要了解二叉搜索树,如果点进来的同学对于二叉搜索树还不熟悉的话,可以先看看我这篇博客:【C++】二叉搜索树。

当然,如果想要更深入的理解map和set的话,肯定是还要学更高级一点的树的,就比如说AVL树、红黑树,但是这里只想要简单用用的话,二叉搜索树先搞清楚就差不多够用了。

本篇主要讲讲map和set的基本用法,最后会有两道题来巩固一下。

后面的两篇博客来说一说关于AVL树的实现和红黑树的实现来更好的理解map和set。

正式开始

STL将容器分为两种。
一种是序列式容器,也就是数据结构中的线性表,包括有vector、list、deque等等。
还有一种是关联式容器,比如我们这里要讲的map和set。关联就是指存放的元素之间是相互关联的,可以通过某一元素来找到其他元素,不像vector那样的,各存各的元素。

那么set和map对应到前一篇的二叉搜索树中分别就是key模型和key/value模型。

先来说set。

还是cplusplus的网站:set,里面的接口我挑着讲。

set

在这里插入图片描述
如上所示,三个模版参数。

  1. T就相当于是key。
  2. compare是仿函数,默认情况下的是less,就是左<根<右。如果传的是greater就是左>根>右。
  3. alloc就是给内存池,这个参数和之前那些容器一样,还没有到该讲的时候,先不讲。

构造

在这里插入图片描述

这三个最常用的就是无参构造。
无参构造初始化了之后就不断插入:
在这里插入图片描述

insert接口如下:
在这里插入图片描述
还是插值、按位置插、迭代器区间。后两个用的都很少。
第一个插值的返回值这里先不细讲等会讲map的时候会说。

如果我们想要遍历的话可以用迭代器:
在这里插入图片描述
这里set的迭代器只支持访问,不支持修改,如果直接*it修改的话会报错。

当然有了迭代器就支持范围for:
在这里插入图片描述

C++11里面还有个用大括号初始化的:
在这里插入图片描述
这个演示一下:
在这里插入图片描述
这里插入的时候会直接去重,然后遍历的时候打印出来是有序的,这也和二叉搜索树的特性是相符的。

迭代器区间构造用到的场景比较少。

在这里插入图片描述

我们可以将compare改为greater的,这样遍历的时候用正向迭代器就变成了降序:

在这里插入图片描述

如果用less但是还想排降序的话,可以用反向迭代器:
在这里插入图片描述

拷贝构造用的也比较少,因为拷贝构造的开销还是比较大的,用的时候要慎重。

还可以用数组来初始化:

在这里插入图片描述

erase

在这里插入图片描述

删除,值、位置、迭代器区间。后两个不常用,演示第一个:
在这里插入图片描述

在这里插入图片描述

find

在这里插入图片描述

find的值如果存在,就返回那个值对应的迭代器。如果不存在,就返回end()。

这里可以用find来查找并删除某值。

在这里插入图片描述

因为可能有值不存在的情况,所以上述的写法是不严谨的,如果删除一个不存在的数就会出问题:
在这里插入图片描述
上面是debug的,如果是release的话就会直接出问题,因为release下会省略assert:
在这里插入图片描述

所以应该这样写:
在这里插入图片描述
判断一下是不是end,是了再删除,不是的话就别删。

这里里比直接erase值更好的一点是,如果某个值不存在,那么就可以打印出该值不存在的信息,如果是直接用erase删除的话,就没法打印了。

在这里插入图片描述
如果不需要打印信息的话可以直接用erase删值。

swap

在这里插入图片描述

两个set的对象交换的时候就用set的swap,因为set的swap就是简单的将根节点的指针交换就行了。不要用算法库中的swap,赋值和拷贝构造的开销会很大。

count

在这里插入图片描述

这个函数就是统计某个值出现的次数,可以说对于set来说没有什么用,因为set中的值永远出现的是一次。

set中有这个函数单纯是为了使得库提供的函数接口的一致性而给的,因为multiset中就有这个count,而multiset就和set的底层一样,但是multiset允许键值冗余,所以可以有多个值同时存在,我们就可以在multiset中使用count。

不过我们可以用count来判断某个值是否存在,如果存在返回1,如果不存在返回0。
在这里插入图片描述

lower_bound 和 upper_bound

lower_bound

在这里插入图片描述
这个函数返回>=val的值的迭代器,如果这个>=的值存在的话,就返回这个值的迭代器,如果不存在的话就返回end。
在这里插入图片描述

在这里插入图片描述

upper_bound

在这里插入图片描述
这个是返回>的值的迭代器。

直接给个和lower_bound一块的示例:
在这里插入图片描述

equal_range

在这里插入图片描述

这里的返回值为pair,这里讲一讲。

pair

在这里插入图片描述
就是一个存放一对值的结构体。

两个值库中定义的名字是first和second。
大概代码如下:

template <class T1, class T2>
struct pair
{
	typedef T1 first_type;
	typedef T2 second_type;
	
	T1 first;
	T2 second;
	
	pair(): first(T1()), second(T2())
	{}
	pair(const T1& a, const T2& b): first(a), second(b)
	{}
};

所以就好说了。

在这里插入图片描述
上面的返回值为pair,而且pair中两个参数的类型是迭代器。也就是返回两个值的迭代器。
返回的是一个左闭右开的区间:[x, y)。(x,y均为迭代器)
如果val的值存在于树中,x=val,y为++x。
如果val不存在,但还在值的范围内,x>val,y=x;不在范围内,x,y都是end()。

在这里插入图片描述

在这里插入图片描述

multiuset

上面也提到了,这个类型支持键值冗余,也就是可以同时存放多个值。

演示一下:
在这里插入图片描述
这里的multiset就是单纯的排序了。没有了去重的功能。

find

这里find的话,如果值没有重复,就返回那个值的迭代器,如果重复了,就返回中序遍历顺序下该值第一次出现的值的迭代器。

在这里插入图片描述

如果想要访问第二次出现的值的话++一下返回的迭代器就好了。

erase

这里如果是删除重复的值的话,比如说x,会把x全部删除。

在这里插入图片描述

和上面给迭代器的删除不一样。

count

这个上面也提过了,这里可以用。

在这里插入图片描述

其他功能就跟set类似了,这里就不继续讲,multiset了。

map

map也分map和multimap,和上面一样,multimap允许键值冗余,而map不允许。

二者都是二叉搜索树中的 key/value 模型。

先来说map。

在这里插入图片描述
这里有四个模版参数。

第一个是键值key,这个值是用来进行关键字的比较的。

第二个是附加项T,这就是一个与key相关的一个值,插入的时候是按照key比较的,而不是T。也就是二叉搜索树中key/value模型中的value。

第三个是仿函数Compare,默认值也是less,不过less只有一个key,也就是比较的时候按照key比较,T不会参与。

第四个是内存池,暂时不讲。里面的那个pair前面也讲过了。

map中有几个内嵌类型(map中typedef的类型)要说一下。
在这里插入图片描述

  1. key_type 模版参数中的第一个参数key
  2. mapped_type 模版参数中的第二个参数T
  3. value_type pair<const key_type,mapped_type>
  4. key_compare 模版参数中的第三个参数Compare

构造

在这里插入图片描述

一般没有直接初始化的时候就给值的,直接调用一下默认构造然后再插入或者用[ ]就行。

所以这里就不讲了。注意一下拷贝构造慎用,消耗较大。

insert

在这里插入图片描述

这里insert,看第一个,返回值先不说,先说参数。
参数为value_type,也就是pair<const key_type,mapped_type>。
那么我们用的时候要这样(下面的例子为写一个字典):
在这里插入图片描述
可以看到,排序的时候是按照ASCII排的。

但是这样有点麻烦了。
可以用匿名对象:
在这里插入图片描述

但是也没有方便到哪去。
我们可以typedef一下pair:
在这里插入图片描述

还有一个接口,专门用来搞pair的。
就是make_pair。

在这里插入图片描述

这个函数的大概实现为:
在这里插入图片描述
其实就是搞了个pair的对象。

用用:
在这里插入图片描述
用的时候可以不像pair那样显示写类型,make_pair是一个函数模版。

上面的都是用范围for来遍历的,下面来说说用迭代器遍历。
在这里插入图片描述
上面可以看到,it解引用得到的是pair类型的对象,然后一个 . 访问其first和second。

如果我们想用指针呢?
在这里插入图片描述
前面list模拟实现的博客中也讲过了,it->返回的是pair类型的指针,如果想要再访问pair中的成员时,就要再加一个->,但是这里为了可读性,编译器做了优化,直接省略掉了一个->。

注意到我上面用范围for的时候里面的赋值为const auto& e的,重在引用,因为map中的存放的为pair类型的数据,如果pair中又存放了string或者其他自定义类型的数据的话,用引用效率就会高不少。

其实上面的insert用的也不多,我只是为了给大家演示演示。
用的最多的还是[ ]。

[ ]

这个才是最需要学的。

如果我们想要用map统计……出现的次数,怎么搞呢?

比如:统计这里面各个水果出现的次数。

string arr[] = { "苹果", "西瓜", "苹果", "西瓜", "苹果", 
		"苹果", "西瓜", "苹果", "香蕉", "苹果", "香蕉" };

两种方法:

  1. 用insert

代码如下:
在这里插入图片描述

  1. 用[ ]

在这里插入图片描述

可以看到,用[ ]非常的方便。

那么[ ]底层不了解的话,上面是很难看懂的,下面就来说说。

[ ]底层原理

在这里插入图片描述
返回值是mapped_type就是第二个模版参数T。也就是对应到pair中的second的类型。

实际的返回值是这样的:
在这里插入图片描述
新手可能一眼就被吓到了。我们来逐个分析分析。

首先里面有个insert,用到了这个函数的返回值。
那么insert返回值刚刚没讲,就是要在这里讲。
在这里插入图片描述
对比一下上面的调用的话,就是第一个插值的insert,返回值为pair<iterator, bool>类型的。

看下库中的解释:
在这里插入图片描述
意思就是上面提供的第一个insert函数返回了一个pair类型的对象(假如说是x),不管是插入了新元素(假如说新插入的元素是y)或是原来map中就有这个元素y(原来有的话就不会插入),x的first都会指向insert之后的map中的元素y,也就是那个元素的迭代器;如果原来map中没有这个新插入的元素y,x的second就是true,代表插入成功了,如果原来map中有这个新插入的元素y,x的second就是false,代表插入失败了。

再看一下函数调用:
在这里插入图片描述
我把insert对应的括号的匹配给画出来了,去掉之后是:
在这里插入图片描述
其返回值是pair对象,然后访问了这个对象的first,first为迭代器,指向新插入的pair,解引用之后又访问了其指向的pair的second,也就是第二个模版参数T。所以最终返回值类型就是第二个模版参数T,对应的就是insert之后的pair的second。

对应到我们上面的countMap[str],其最终的返回值就是striing对象所对应的int。

所以当我们第一次用[ ]的时候,是没有str的,先插入了,然后再让对应的int++了一下。

再看一下上面调用的insert,调用时传参为make_pair(k,mapped_type()),也就是说second会调用其默认构造函数,而上面的int默认值就是0。所以插入之后为0,返回之后再++正好为1,所以插入一次就加一。就达到了统计次数的目的。

如果感觉上面的[ ]重载实现的太难以理解了,我们也可以自己来实现一下:

在这里插入图片描述

再总结一下[ ]的返回值:

  1. map中有这个key,返回的是T的引用。即查找+修改T的功能。
  2. map中没有这个key,插入一个pair(key, T())并返回T的引用。 即插入+修改T的功能。

现在我们再来写刚刚写过的词典的话就非常简单了。
在这里插入图片描述
在这里插入图片描述

map中还有个at,也可以返回T的引用,和[ ]不一样的是,at不在的话是抛异常:
在这里插入图片描述

可以看到将map的时候有一大堆的pair,可以说pair就是专门为[ ]准备的。

再来说一点关于multimap的

multimap

就说一点,允许键值冗余。
如果允许键值冗余的话,就不能用[ ]和at了,因为如果有重复的元素,编译器不知道该匹配哪个,直接报错:
在这里插入图片描述

但是好处就是我们可以插入相同的K值了。
在这里插入图片描述
left有左边,也有剩余的意思。可以看到,插入了两次左边,一次剩余。

剩下的就不说了。

下面来两道题来练练手:

两道题目

两道题目链接:
前K个高频单词
两个数组的交集

前K个高频单词

在这里插入图片描述

这道题乍一看是topK问题,有的同学就想到了用堆来写。确实可以。

法一:优先级队列

我们先用map来统计一下各个单词出现的次数。然后再根据出现的次数来将map中的各个pair放到优先级队列(大堆)中,然后就可以不断获取堆顶元素来获取topK个单词。

但是题目中要求,各个单词要按照出现频率来排序,当两个单词出现的频率相同时,按照字典序排序,也就是按照ASCII来排。

最终实现的代码如下:
在这里插入图片描述
用到仿函数的时候,里面的逻辑有点绕。

法二:数组排序

我们先用map统计次数,此时map中的各个pair的顺序是按照string来排的。然后我们将map中的各个pair放入到数组中,用数组按照pair的second来进行排序。同样也是要写仿函数来控制频率相同的情况。

实现代码如下:
在这里插入图片描述

方法三:两个map

两个map,一个用来统计出现次数,一个用来根据次数排序。

在这里插入图片描述

两个数组的交集

这道题其实可以直接用双指针,但用双指针的话还要先排序再去重,太麻烦了,我们可以直接用set。

求交集有一个思路。在两个数组有序且不重复的前提下,比如说:
在这里插入图片描述
分为上下两个数组。

每个都从头开始遍历:
在这里插入图片描述

因为是有序的,所以当某个值 it1 小比另一个值 it2 小的时候,那么 it1 一定比另一个数组中的所有数都小。it2同理。所以小的时候就让其++,然后再对比,等的时候就同时++。当一个走到尽头时,就说明没有交集可找了,此时停下来就行。

因为需要排序 + 去重,而set正好能够满足这个条件,代码如下:
在这里插入图片描述

同样的思路,我们可以用来求差集。
在这里插入图片描述

就讲到这。

到此结束。。。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/845021.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[golang gin框架] 45.Gin商城项目-微服务实战之后台Rbac微服务之角色权限关联

角色和权限的关联关系在前面文章中有讲解,见[golang gin框架] 14.Gin 商城项目-RBAC管理之角色和权限关联,角色授权,在这里通过微服务来实现角色对权限的授权操作,这里要实现的有两个功能,一个是进入授权,另一个是,授权提交操作,页面如下: 一.实现后台权限管理Rbac之角色权限关…

pom文件---maven

027-Maven 命令行-实验四-生成 Web 工程-执行生成_ev_哔哩哔哩_bilibili 27节.后续补充 一.maven下载安装及配置 1)maven下载 2) settings文件配置本地仓库 3)settings配置远程仓库地址 4)配置maven工程的基础JDK版本 5)确认JDK环境变量配置没问题,配置maven的环境变量 验证…

C++ 派生类成员的标识与访问——虚基类

当某类的部分或者全部直接基类是从另一个共同的基类派生而来时&#xff0c;在这些直接基类中从上一级共同基类继承来的成员就拥有相同的名称。在派生类的对象中&#xff0c;这些同名数据成员在内存中同时多个副本&#xff0c;同一个函数名会有多个映射。 可以通过作用域分辨符…

CISCO MDS 9148 SAN Switch 交换机命令配置方法:

前言 CISCO MDS 9148 SAN 交换机已经停产&#xff0c;但还是要掌握一下配置的方法&#xff1a; 升级款后面 9148S 或者 9100系列&#xff0c;但配置方式基本都差不多&#xff0c;掌握一个就好&#xff1a; 高性能和极具吸引力的价值 Cisco MDS 9148S 16G 多层光纤交换机是下…

24届近5年南京理工大学自动化考研院校分析

今天学长给大家带来的是南京理工大学控制考研分析 满满干货&#xff5e;还不快快点赞收藏 一、南京理工大学 ​ 学校简介 南京理工大学是隶属于工业和信息化部的全国重点大学&#xff0c;学校由创建于1953年的新中国军工科技最高学府——中国人民解放军军事工程学院&#xf…

用户权限提升Sudo

目录 前言 一、su的用法 二、sudo提权 总结 前言 sudo是linux系统管理指令&#xff0c;是允许系统管理员让普通用户执行一些或者全部的root命令的一个工具&#xff0c;如halt&#xff0c;reboot&#xff0c;su等等。换句话说通过此命令可以让非root的用户运行只有root才有权限…

vue 新学习 06 js的prototype ,export暴露,vue组件,一个重要的内置关系

部分内容参考的这篇文章 原文链接&#xff1a;https://blog.csdn.net/harry5508/article/details/84025146 写的很好。 01 在js中&#xff1a; 原型链 注意&#xff1a;构造函数.prototype实例化对象.__proto__&#xff0c;都是指向函数的原型。 export&#xff1a; -export用…

品牌宣传与媒体传播是声誉管理的主要方式之一

企业声誉是现如今影响品牌信任度、客户忠诚度的重要因素&#xff0c;也被视为企业的一种无形资&#xff0c;更影响着企业未来的发展。因此&#xff0c;企业声誉管理也日渐成为企业管理的重要课题之一&#xff0c;尤其在品牌营销管理领域。 什么是声誉管理&#xff1f;声誉管理有…

【果树农药喷洒机器人】Part1:研究现状分析以及技术路线介绍

本专栏介绍&#xff1a;付费专栏&#xff0c;持续更新机器人实战项目&#xff0c;欢迎各位订阅关注。 关注我&#xff0c;带你了解更多关于机器人、嵌入式、人工智能等方面的优质文章&#xff01; 文章目录 一、项目背景二、国内外研究现状2.1 国内研究现状2.2 国外研究现状 三…

回归预测 | MATLAB实现POA-CNN-BiGRU鹈鹕算法优化卷积双向门控循环单元多输入单输出回归预测

回归预测 | MATLAB实现POA-CNN-BiGRU鹈鹕算法优化卷积双向门控循环单元多输入单输出回归预测 目录 回归预测 | MATLAB实现POA-CNN-BiGRU鹈鹕算法优化卷积双向门控循环单元多输入单输出回归预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 MATLAB实现POA-CNN-BiGRU鹈鹕…

kretprobe 和 fexit

kretprobe 孬&#xff0c;跟朋友简单讨论了相关主题&#xff0c;发现 fexit 高尚。 fexit 的把戏在 2020 年中那段走火入魔的时间玩过不少&#xff0c;没想到就是 fexit 的标准&#xff0c;看来多数人觉得正确的思路它就是正确的。 kretprobe 每次调用函数都要执行复杂的带锁…

从零构建深度学习推理框架-6 构建计算图

PNNX PNNX项目 PyTorch Neural Network eXchange&#xff08;PNNX&#xff09;是PyTorch模型互操作性的开放标准。PNNX为PyTorch提供了一种开源的模型格式&#xff0c;它定义了与Pytorch相匹配的数据流图和运算图&#xff0c;我们的框架在PNNX之上封装了一层更加易用和简单的计…

外网通过ipv6访问家里设备

想从公司访问家里的设备&#xff0c;比较轻松方便的&#xff0c;用向日葵也可以远程。但是家里电脑比较old的了&#xff0c;向日葵开起来&#xff0c;占用内存挺大的&#xff0c;想尝试windows自带的“mstsc”&#xff0c;所以硬着头皮搞ipv6. &#xff08;重点提示&#xff1…

干草垛(Haystack)里找“`膝尖儿`(Kneedle)”: 算法复现

干草垛(Haystack)里找“膝尖儿(Kneedle)”: 算法复现 缘起 源 引用: Finding a “Kneedle” in a Haystack: Detecting Knee Points in System Behavior Ville Satopa † , Jeannie Albrecht† , David Irwin‡ , and Barath Raghavan †Williams College, Williamstown, MA …

Connection reset原因分析及解决思路

Connection reset原因分析及解决思路 我们在开发过程中经常会出现Connection reset问题&#xff0c;包括http调用&#xff0c;数据库连接等场景。出现Connection reset的原因很多&#xff0c;本文从tcp层面简单介绍下Connection reset出现的原因和问题&#xff0c;以及在实际开…

AJAX-笔记(持续更新中)

文章目录 Day1 Ajax入门1.AJAX概念和axios的使用2. 认识URL3.URL的查询参数4.常用的请求方法和数据提交5.HTTP协议-报文6.接口文档7.form-serialize插件8.案例用户登录 Day2 Ajax综合案bootstrap弹框图书管理图片上传更换背景个人信息设置 Day3 AJAX原理XMLHttpRequestPromise封…

【雕爷学编程】Arduino动手做(193)---移远 BC20 NB+GNSS模块7

37款传感器与模块的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&#x…

springboot集成分布式任务调度系统xxl-job(调度器和执行器)

一、部署xxl-job服务端 下载xxl-job源码 下载地址&#xff1a; https://gitee.com/xuxueli0323/xxl-job 二、导入项目、创建xxl_job数据库、修改配置文件为自己的数据库 三、启动项目、访问首页 访问地址&#xff1a; http://localhost:8080/xxl-job-admin/ 账号&#xff1…

IO进程线程day9(2023.8.7)

一、Xmind整理&#xff1a; 消息队列的原理&#xff1a; 共享内存的原理&#xff1a; 二、课上练习&#xff1a; 练习1&#xff1a;用信号的方式回收僵尸进程&#xff08;重点&#xff01;&#xff09; 1.子进程退出后&#xff0c;父进程会收到17)SIGCHLD信号。 2.父进程中捕获…

04-1_Qt 5.9 C++开发指南_常用界面设计组件_字符串QString

本章主要介绍Qt中的常用界面设计组件&#xff0c;因为更多的是涉及如何使用&#xff0c;因此会强调使用&#xff0c;也就是更多针对实例&#xff0c;而对于一些细节问题&#xff0c;需要参考《Qt5.9 c开发指南》进行学习。 文章目录 1. 字符串与普通转换、进制转换1.1 可视化U…