【C++】map和set用法详解
文章目录
- 1.关联式容器
- 2.键值对
- 3.树形结构的关联式容器
- 3.1 set
- 3.1.1 set的介绍
- 3.1.2 set的模板参数列表
- 3.1.3 set的使用
- 3.2 map
- map的介绍
- map的模板参数列表
- map的使用
- 关于map的元素访问总结
- 3.3multimap
1.关联式容器
我们接触过STL中的部分容器,比如:vector,list,deque,forward_list(C++11)等,这些容器统称为序列式容器,因为其底层为线性序列的数据结构,里面存储的是元素本身。那么什么是关联式容器?它与序列式容器有什么区别?
关联式容器也是用来存储数据的,与序列式容器不同的是,其里面存储的是<key,value>结构的键值对,在数据检索时比序列式容器效率更高。
2.键值对
用来表示具有一一对应关系的一种结构,该结构中一般只包含两个成员变量key和value,key代表键值,value表示与key对应的信息。比如:现在要建立一个英汉互译的字典,那该字典中必然有英文单词与其对应的中文含义,而且,英文单词与其中文含义是一一对应的关系,即通过该单词,在词典就可以找到与其对应的中文含义;比如每个学生的学号跟他的姓名,年龄,专业(类型可定义成数组)存在对应关系,找到学号,就可以查到这个学生的相关信息。
SGI-STL中关于键值对的定义:
//STL中关于键值对的定义
template<class T1,class T2>
struct pair
{typedef T1 first_type;typedef T2 first_type;T1 first;T2 second;//构造函数初始化pair():first(T1())//key,second(T2())//value{}//拷贝构造函数,使用实例pair<string,int>pair(const T1& a,const T2& b):first(a),second(b){}
};
3.树形结构的关联式容器
根据应用场景不同,STL总共实现了两种不同结构的关联式容器:树形结构与哈希结构。树形结构的关联式容器主要有四种:map,set,multimap,multiset。这四种容器的共同点是:使用平衡二叉树(即红黑树)作为底层,容器中的元素是一个有序的序列。下面依次介绍每一个容器。
3.1 set
3.1.1 set的介绍
1.set翻译为集合,是一个内部自动有序且不含重复元素的关联式容器。当我们想要去重,就可以利用到set,是一个很直观的接口,并且加入set之后可以实现自动排序,需要注意的是set的元素默认按照小于比较。
2、在set中,每个value必须是唯一的。set中的元素不能在容器中修改(元素总是const),但是可以在容器中插入或删除数据。一般的做法是先删除旧元素,然后添加新元素,这当然是为了维护里面元素的有序性。
3.set在底层使用二叉搜索树(红黑树)实现的。
4.与map/multimap不同,map/multimap中存储的是真正的键值对<key,value>,而set中只放value,但在底层实际存放的是由<value,value>构成的键值对。
5.在set中查找某个元素,时间复杂度为:log2N
3.1.2 set的模板参数列表
//头文件
#include<set>
std::set
template<class T,class Compare = less<T>,class Alloc = allocator<T>>
T:set中存放元素的类型,实际在底层存储<value, value>的键值对。
Compare(仿函数):set中元素默认按照小于来比较
Alloc:set中元素空间的管理方式,使用STL提供的空间配置器管理
3.1.3 set的使用
点此进入->set的文档
我在下列代码一一解释了关于set常见的函数该如何使用,
#include<iostream>
#include<set>
using namespace std;
int main()
{set<int> myset;//定义//迭代器的使用set<int>::iterator itlow, itup;for (int i = 1; i < 10; i++){//插入数据//注意set会将元素自动排序myset.insert(i * 10);//10 20 30 40 50 60 70 80 90}//删除35-75之间的数据itlow = myset.lower_bound(35);//返回第一个大于等于key_value的定位器itup = myset.upper_bound(75);//返回最后一个小于等于key_value的定位器//注意:set中的删除操作不进行错误检查,所以用的时候最好用一下find()函数,//返回给定值定位器,如果没找到则返回end()。myset.erase(itlow, itup);/*for (auto it = myset.begin(); it != myset.end(); ++it){cout << *it <<' ';}*///这里可以使用范围for,需要加&,因为拷贝也是存在一定代价的,不需要修改的话,也尽量加const//范围for的底层是迭代器,操作由编译器完成for (auto& s : myset){cout << s << ' ';}cout << endl;//count() 用来查找set中某个键值出现的次数。这个函数在set并不是很实用,在map中的用处比较大//因为一个键值在set只可能出现0或1次,==这样就变成了判断某一键值是否在set出现过==。cout << myset.count(100) << endl;//0不存在cout << myset.count(30) << endl;//1存在return 0;
}
3.2 map
map的介绍
点此进入->map的官方文档
1.map是关联式容器,它按照特定的次序(按照key来排序),存储由key和value组合而成的元素。
2.map中,是真正的键值对<key,value>,键值key和值value的类型可能不同,并且在map内部,key与value通过成员类型value_type绑定在一起,取名为pair。
typedef pair<const key,T> value_type
- 在内部,map中的元素总是按照键值key进行比较排序的。
- map支持下标访问,即在[]中放入key,就可以找到与key对应的value。在map中key不允许修改,key对应的value可以修改。
- map通常被实现为二叉搜索树(更准确的说:平衡二叉搜索树(红黑树))。
map的模板参数列表
相比set,map多了一个参数
#include<map>
std::map
template<class key,class T,class Compare = less<key>,class Alloc = allocator<pair<const Key,T>>
key: 键值对中key的类型
T: 键值对中value的类型
Compare: 比较器的类型,map中的元素是按照key来比较的,缺省情况下按照小于来比
较
Alloc:通过空间配置器来申请底层空间,不需要用户传递,除非用户不想使用标准库提供的
空间配置器
注意:在使用map时,需要包含头文件。
map的使用
map的官方文档已包括所有关于map的接口函数,现在我们来尝试使用它的常用接口。
最重要的应该有:插入函数insert,[ ]下标访问操作符,对于pair的理解,迭代器的使用,删除函数。
#include<iostream>
#include<map>
#include<string>
using namespace std;
int main()
{//定义一个map对象map<int, string> Stu;//1.用insert函数插入pairStu.insert(pair<int, string>(01, "张三"));Stu.insert(pair<int, string>(02, "李四"));Stu.insert(pair<int, string>(03, "王五"));//但是我们发现在这里,写一个pair很不方便,因此我们可以用make_pair//make_pair的底层:是一个函数模板,还是调用pair去构造/*template<class T1,class T2>pair<T1,T2> make_pair(T1 x,T2 y){return (pair<T1, T2>(x, y));}*/Stu.insert(make_pair(04, "李芳"));//3.第三种用[]的方式插入数据Stu[123] = "王五";//4.map的迭代器//map<int, string>::iterator it = Stu.begin();auto it = Stu.begin();while (it != Stu.end()){cout << it->first << ":" << it->second << endl;++it;}//最好加上引用,如果不修改的话,把const加上更好for (const auto& kv : Stu){cout << kv.first << ":"<< kv.second << endl;}cout << endl;return 0;
}
活学活用:统计水果出现的次数
//利用map,统计水果出现的次数string arr[] = { "苹果", "西瓜", "香蕉", "草莓", "苹果", "西瓜", "苹果", "苹果", "西瓜", "苹果", "香蕉", "苹果", "香蕉" };map<string, int> countMap;for (auto& e: arr){//map<string, int>::iterator it = countMap.find(e);auto it = countMap.find(e);//find函数找不到则返回end()if (it == countMap.end()){countMap.insert(make_pair(e,1));}else{it->second++;}}//有个更简洁的方法//for (auto& e : arr)//{// countMap[e]++;//}for (const auto& e : countMap){cout << e.first << ":" << e.second;}
关于map的元素访问总结
- map中的元素是键值对,我们需要学习pair,然后学会使用map容器,插入数据,管理数据。
- map中的key是唯一的,并且不能修改。
- map默认按照小于(升序)的方式,并且是对key排序的。map中的元素如果用迭代器去遍历,是采用中序遍历的方式,可以得到一个有序序列。
- map的底层是一个平衡二叉树(红黑树),查找效率很高O(logN)。
- map中[ ]下标访问操作符是一个大头,它支持查找,修改,插入。operator[ ]向map中插入元素的原理:用<key,T()>构造一个键值对,然后调用insert()函数将该键值对插入到map中,map中的键值对key一定是唯一的,如果key已经存在,插入失败,insert函数返回该key所在位置的迭代器。如果key不存在,插入成功,insert函数返回新插入元素所在位置的迭代器。
- operator[ ]函数最后将insert返回值键值对中的value返回。(也就是pair类中的second成员)
- 注意:在元素访问时,有一个与operat[ ]类似的函数,是at()函数,但是这个函数不常用,它们都是通过key找到与key对应的value然后返回其引用,不同的是:当key不存在时,operator[]会用默认的pair插入,返回该默认的value,而at()函数会直接抛异常。
- operator[]底层实现如下->
3.3multimap
注意:multimap和map的唯一不同就是:map中的key是唯一的,而multimap中key是可以重复的。
multimap中的接口可以参考map,功能都是类似的。
注意:
- multimap中的key是可以重复的。
- multimap中的元素默认将key按照小于来比较。
- multimap中没有重载operator[]操作。
- 使用时与map包含的头文件相同。
相关文章:

【C++】map和set用法详解
文章目录1.关联式容器2.键值对3.树形结构的关联式容器3.1 set3.1.1 set的介绍3.1.2 set的模板参数列表3.1.3 set的使用3.2 mapmap的介绍map的模板参数列表map的使用关于map的元素访问总结3.3multimap1.关联式容器 我们接触过STL中的部分容器,比如:vecto…...

BLIP2-图像文本预训练
文章目录摘要解决问题算法模型结构通过frozen图像编码器学习视觉语言表征图像文本对比学习(ITC)基于图像文本生成(ITG)图文匹配(ITM)从大规模语言模型学习视觉到语言生成模型预训练预训练数据预训练图像编码…...

Faster-Rcnn修改转数据集文件
目录 学习python的一些基础知识 argparser assert关键字 让你秒懂Python 类特殊方法__getitem__ lxml.etree.fromstring的使用 统计一下json文件内的种类 正脸红外光 正脸-混合红外光 正脸-交叉偏振光 正脸-平行偏振光 正脸-紫外光 正脸-棕色光 调用mydataset可视化…...

带你沉浸式体验删库跑路
前言:学习的过程比较枯燥,后面会记录一些比较有意思的东西,比如程序员之间流传的删库跑路的梗,当然本次测试是在虚拟机上进行的并进行了快照保护,所以其实没太大问题。首先得要有一个虚拟机要有一个linux iso文件装在虚拟机上以上两点不是本文重点,如果有需要可以私…...

Linux学习(8.5)文件内容查阅
目录 文件内容查阅: 直接检视文件内容 cat (concatenate) tac (反向列示) nl (添加行号列印) 可翻页检视 more (一页一页翻动) less (一页一页翻动) 数据撷取 tail (取出后面几行) 非纯文字档: od 修改文件时间或建置新档: touc…...

【Docker】命令总结
目录 1.镜像命令 1.1拉取镜像 1.2查看镜像 1.3保存镜像 1.4导入镜像 2.容器命令 2.1创建并运行容器 2.2删除容器 2.3进入容器 2.4查看容器状态 2.5暂停容器 2.6恢复容器 2.7停止容器 2.8启动容器 2.8查看容器日志 3.数据卷命令 3.1创建数据卷 3.2查看所有数据…...

并发编程-学习总结(上)
目录 1、线程基础 1.1、线程实现方法 1.2、如何正确停止线程 1.3、Java线程的六种状态 1.4、wait/notify/notifyAll注意事项 1.4.1、为什么 wait 、notify、notifyAll必须在 synchronized 保护的同步代码中使用? 1.4.2、为什么 wait/notify/notifyAll 被定义…...

QT之OpenGL混合
QT之OpenGL混合1. 概述2. 实现2.1 丢弃片段2.1.1 Demo2.2 混合2.2.1 相关函数2.2.2 排序问题2.2.3 Demo1. 概述 OpenGL中,混合(Blending)通常是实现物体透明度(Transparency)的一种技术。 2. 实现 2.1 丢弃片段 在某些情况下,有些片段是只需要设置显…...
【1255. 得分最高的单词集合】
来源:力扣(LeetCode) 描述: 你将会得到一份单词表 words,一个字母表 letters (可能会有重复字母),以及每个字母对应的得分情况表 score。 请你帮忙计算玩家在单词拼写游戏中所能获…...

nginx模块介绍
新编译前,在对应的nginx原编译文件夹 如:nginx-1.23.0 下,要 make clean 清空以前编译的objs文件夹,实际上就是执行了rm objs文件夹。 很多要用到git,先yum install git -y echo-nginx-module 让nginx直接使用echo的…...

排错工具ping和trace(电子科技大学TCP/IP实验四)
一.实验目的 1、了解网络连通性测试的方法和工作原理 2、了解网络路径跟踪的方法和工作原理 3、掌握 MTU 的概念和 IP 分片操作 4、掌握 IP 分组生存时间(TTL)的含义和作用 5、掌握路由表的作用和路由查找算法 二.预备知识 …...
node.js中ws模块创建服务端和客户端
一、WebSocket出现的原因 1、Http协议发布REST API 的不足: 每次请求响应完成之后,服务器与客户端之间的连接就断开了,如果客户端想要继续获取服务器的消息,必须再次向服务器发起请 求。这显然无法适应对实时通信有高要求的场景…...
kubernates-1.26.1 kubeadm containerd 单机部署
k8s1.26 kubeadm containerd 安装 kubeadm init 时提示 containerd 错误 failed to pull image “k8s.gcr.io/pause:3.6” 报错日志显示containerd pull时找不到对应的pause版本,而不是registry.k8s.io/pause:3.9 [rootk8s-master containerd]# kubeadm init --k…...

如何在 iPhone 上恢复已删除的通话记录/通话记录
您的通话记录/通话记录可能很重要,尤其是当您想要拨打之前联系过但未保存的号码时。如果您碰巧删除了通话记录(有意或无意),本指南将帮助您了解如何检索它们并找回您需要使用的所有记录。我们将根据您的情况和您拥有的工具讨论不同…...

Canonical为所有支持的Ubuntu LTS系统发布了新的Linux内核更新
导读Canonical近日为所有支持的Ubuntu LTS系统发布了新的Linux内核更新,以解决总共19个安全漏洞。新的Ubuntu内核更新仅适用于长期支持的Ubuntu系统,包括Ubuntu 22.04 LTS(Jammy Jellyfish)、Ubuntu 20.04 LTS(Focal F…...

MS9122是一款USB单芯片投屏器,内部集成了USB2 0 控制器和数据收发模块、HDMI 数据接口和音视频处理模块。MS9122可以通过USB接口显示
MS9122是一款USB单芯片投屏器,内部集成了USB2.0 控制器和数据收发模块、HDMI 数据接口和音视频处理模块。MS9122可以通过USB接口显示或者扩展PC、智能手机、平板电脑的显示信息到更大尺寸的显示设备,支持HDMI视频接口。 主要功能特征 HDMI v1.4兼容 最大…...
C++学习笔记-数据抽象
简单的说,数据抽象是用来描述数据结构的。数据抽象就是 ADT。一个 ADT 主要表现为它支持的一些操作,比方说 stack.push、stack.pop,这些操作应该具有明确的时间和空间复杂度。另外,一个 ADT 可以隐藏其实现细节,比方说…...

【Android】Android开发笔记(一)
【Android】Android开发笔记(一) 在Android Studio中import module和delete moduleimport moduledelete moduleAndroid Studio中App(Module)无法正常运行在实机上测试App一些基本概念App的工程结构结语在Android Studio中import m…...

C语言数据结构(二)—— 受限线性表 【栈(Stack)、队列(Queue)】
在数据结构逻辑层次上细分,线性表可分为一般线性表和受限线性表。一般线性表也就是我们通常所说的“线性表”,可以自由的删除或添加结点。受限线性表主要包括栈和队列,受限表示对结点的操作受限制。一般线性表详解,请参考文章&…...

线程安全之synchronized和volatile
目录 1.线程不安全的原因 2.synchronized和volatile 2.1 synchronized 2.1.1 synchornized的特性 2.1.2 synchronized使用示例 2.2 volatile 我们先来看一段代码: 分析以上代码,t1和t2这两个线程的任务都是分别将count这个变量自增5000次ÿ…...
【SpringBoot】100、SpringBoot中使用自定义注解+AOP实现参数自动解密
在实际项目中,用户注册、登录、修改密码等操作,都涉及到参数传输安全问题。所以我们需要在前端对账户、密码等敏感信息加密传输,在后端接收到数据后能自动解密。 1、引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId...

深入理解JavaScript设计模式之单例模式
目录 什么是单例模式为什么需要单例模式常见应用场景包括 单例模式实现透明单例模式实现不透明单例模式用代理实现单例模式javaScript中的单例模式使用命名空间使用闭包封装私有变量 惰性单例通用的惰性单例 结语 什么是单例模式 单例模式(Singleton Pattern&#…...

【机器视觉】单目测距——运动结构恢复
ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛…...
在 Nginx Stream 层“改写”MQTT ngx_stream_mqtt_filter_module
1、为什么要修改 CONNECT 报文? 多租户隔离:自动为接入设备追加租户前缀,后端按 ClientID 拆分队列。零代码鉴权:将入站用户名替换为 OAuth Access-Token,后端 Broker 统一校验。灰度发布:根据 IP/地理位写…...
MySQL用户和授权
开放MySQL白名单 可以通过iptables-save命令确认对应客户端ip是否可以访问MySQL服务: test: # iptables-save | grep 3306 -A mp_srv_whitelist -s 172.16.14.102/32 -p tcp -m tcp --dport 3306 -j ACCEPT -A mp_srv_whitelist -s 172.16.4.16/32 -p tcp -m tcp -…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...

[ACTF2020 新生赛]Include 1(php://filter伪协议)
题目 做法 启动靶机,点进去 点进去 查看URL,有 ?fileflag.php说明存在文件包含,原理是php://filter 协议 当它与包含函数结合时,php://filter流会被当作php文件执行。 用php://filter加编码,能让PHP把文件内容…...

保姆级【快数学会Android端“动画“】+ 实现补间动画和逐帧动画!!!
目录 补间动画 1.创建资源文件夹 2.设置文件夹类型 3.创建.xml文件 4.样式设计 5.动画设置 6.动画的实现 内容拓展 7.在原基础上继续添加.xml文件 8.xml代码编写 (1)rotate_anim (2)scale_anim (3)translate_anim 9.MainActivity.java代码汇总 10.效果展示 逐帧…...

自然语言处理——文本分类
文本分类 传统机器学习方法文本表示向量空间模型 特征选择文档频率互信息信息增益(IG) 分类器设计贝叶斯理论:线性判别函数 文本分类性能评估P-R曲线ROC曲线 将文本文档或句子分类为预定义的类或类别, 有单标签多类别文本分类和多…...
React父子组件通信:Props怎么用?如何从父组件向子组件传递数据?
系列回顾: 在上一篇《React核心概念:State是什么?》中,我们学习了如何使用useState让一个组件拥有自己的内部数据(State),并通过一个计数器案例,实现了组件的自我更新。这很棒&#…...