C++入门14——set与map的使用

在本专栏的往期文章中,我们已经学习了STL的部分容器,如vector、list、stack、queue等,这些容器统称为序列式容器,因为其底层是线性序列的数据结构,里面存储的是元素本身。而本篇文章我们要来认识一下关联式容器。

🥇关联式容器

在介绍关联式容器之前,先来回顾一下已经学习的数据结构9——二叉搜索树,在这篇文章中,我们认识了什么是K模型和KV模型:

K模型:只有Key作为关键码,结构中只需要存储Key,关键码即为需要搜索到的值。

KV模型:每一个关键码key,都有与之对应的值Value,即的键值对。

有了这个前提,我们就可以认为:

关联式容器就是KV模型的一种应用,关联式容器也是用来存储数据的,与序列式容器不同的是,其里面存储的是结构的键值对,在数据检索时比序列式容器效率更高。

根据应用场景的不同,STL一共实现了两种不同结构的管理式容器:树型结构与哈希结构。树型结 构的关联式容器主要有四种:map、set、multimap、multiset。这四种容器的共同点是:使用平衡搜索树(平衡搜索树是一种二叉搜索树)作为其底层结果,容器中的元素是一个有序的序列。

🥇set

🥈set的官网介绍

1. set是按照一定次序存储元素的容器;

2. 在set中,元素的value也标识它(value就是key,类型为T),并且每个value必须是唯一的。 set中的元素不能在容器中修改(元素总是const),但是可以从容器中插入或删除它们;

3. 在内部,set中的元素总是按照其内部比较对象(类型比较)所指示的特定严格弱排序准则进行排序;

4. set容器通过key访问单个元素的速度通常比unordered_set容器慢,但它们允许根据顺序对 子集进行直接迭代;

5. set在底层是用二叉搜索树(红黑树)实现的。

🥈set的使用

🥉插入

int main()
{set<int> s;//1.插入s.insert(8);s.insert(1);s.insert(7);s.insert(13);s.insert(6);s.insert(3);s.insert(5);s.insert(9);s.insert(14);s.insert(9);return 0;
}

🥉遍历

int main()
{set<int> s;//1.插入s.insert(8);s.insert(1);s.insert(7);s.insert(13);s.insert(6);s.insert(3);s.insert(5);s.insert(9);s.insert(14);s.insert(9);//2.迭代器遍历set<int>::iterator it = s.begin();while (it != s.end()){cout << *it << " ";++it;}cout << endl;//3.范围for遍历for (auto e : s){cout << e << " ";}cout << endl;return 0;
}

🥉查找

//4.查找set<int>::iterator pos = s.find(14);if (pos != s.end()){cout << "已找到" << endl;}

🥉删除

//5.删除//①   用此方法删除,目标值存在就删除,不存在不做任何处理s.erase(2);s.erase(1);for (auto e : s){cout << e << " ";}cout << endl;//②   用此方法删除,目标值存在就删除,不存在会程序报错pos = s.find(8);s.erase(pos);for (auto e : s){cout << e << " ";}cout << endl;pos = s.find(1);s.erase(pos);

🥉lower_bound与upper_bound

//6.lower_bound与upper_bound	3 5 6 7 9 13 14auto start = s.lower_bound(3);  //lower_bound返回的值:>=valcout << *start << endl;auto finish = s.upper_bound(7); //upper_bound的返回值:>valcout << *finish << endl;//找区间[3,7]//while (start != finish)//{//	cout << *start << " ";//	++start;//}//cout << endl;//删除区间[3,7]s.erase(start, finish);for (auto e : s){cout << e << " ";}cout << endl;

🥈set小结

①与map/multimap不同,map/multimap中存储的是真正的键值对<key,value>,set中只放 value,但在底层实际存放的是由<value,value>构成的键值对;

②set中插入元素时,只需要插入value即可,不需要构造键值对;

③set中的元素不可以重复(因此可以使用set进行去重);

④使用set的迭代器遍历set中的元素,可以得到有序序列;

⑤set中的元素默认按照小于来比较;

⑥set中查找某个元素,时间复杂度为:log_2 n;

⑦set中的元素不允许修改(因为修改之后不能保证二叉搜索树的有序性);

⑧set中的底层使用二叉搜索树(红黑树)来实现。

🥇multiset

🥈multiset的官网介绍

1. multiset是按照特定顺序存储元素的容器,其中元素是可以重复的;

2. 在multiset中,元素的value也会识别它(因为multiset中本身存储的就是<value,value>组成的键值对,因此value本身就是key,key就是value,类型为T),multiset元素的值不能在容器中进行修改(因为元素总是const的),但可以从容器中插入或删除; 

3. 在内部,multiset中的元素总是按照其内部比较规则(类型比较)所指示的特定严格弱排序准则进行排序;

4. multiset容器通过key访问单个元素的速度通常比unordered_multiset容器慢,但当使用迭代器遍历时会得到一个有序序列;

5. multiset底层结构为二叉搜索树(红黑树)。

🥈multiset的使用

multiset的使用与set的使用大致相同,下面演示区别较大的地方:

🥉插入

int main()
{multiset<int> s;//1.插入s.insert(8);s.insert(1);s.insert(7);s.insert(13);s.insert(6);s.insert(3);s.insert(5);s.insert(9);s.insert(14);s.insert(9);s.insert(9);s.insert(9);s.insert(9);multiset<int>::iterator it = s.begin();while (it != s.end()){cout << *it << " ";++it;}cout << endl;return 0;
}

也就是说,set用来去重+排序,multiset用来排序

🥉计数

//序列有几个9
cout << s.count(9) << endl;

🥉查找

 当序列中有多个重复的值时,multiset的find返回中序遍历的第一个数!

//3.查找it = s.find(9);while (it != s.end() && *it == 9){cout << *it << " ";++it;}cout << endl;

 

🥈multiset小结

①multiset中在底层中存储的是的键值对;

②mtltiset的插入接口中只需要插入即可;

③与set的区别是,multiset中的元素可以重复,set中value是唯一的;

④使用迭代器对multiset中的元素进行遍历,可以得到有序的序列;

⑤multiset中的元素不能修改;

⑥在multiset中找某个元素,时间复杂度为O(log_2 N);

⑦multiset的作用:可以对元素进行排序。

🥇map

🥈map的官网介绍

1. map是关联式容器,它是按照特定的次序(按照key来比较)存储,是由键值key和值value组合而成的元素;

2. 在map中,键值key通常用于排序和唯一地标识元素,而值value中存储与此键值key关联的内容。键值key和值value的类型可能不同,并且在map的内部,key与value通过成员类型 value_type绑定在一起,为其取别名称为pair: typedef pair value_type;

3. 在内部,map中的元素总是按照键值key进行比较排序的;

4. map中通过键值访问单个元素的速度通常比unordered_map容器慢,但map允许根据顺序对元素进行直接迭代(即对map中的元素进行迭代时,可以得到一个有序的序列);

5. map支持下标访问符,即在[ ]中放入key,就可以找到与key对应的value;

6. map通常被实现为二叉搜索树(更准确的说:平衡二叉搜索树(红黑树))。

🥈map的使用

🥉插入

//1.插入map<string, string> dict;dict.insert(pair<string, string>("pear", "梨"));//pair<string, string> kv("peach", "桃子");pair<string, string> kv = { "peach", "桃子" };dict.insert(kv);//C++11 多参数隐式类型转换(构造函数)dict.insert({ "apple", "苹果" });//C++98dict.insert(make_pair("banana", "香蕉"));dict.insert(make_pair("banana", "香蕉2"));

🥉遍历

//2.遍历//①迭代器遍历map<string, string>::iterator it = dict.begin();while (it != dict.end()){//cout << *it << endl;//cout << (*it).first << (*it).second << endl;cout << it->first << it->second << endl;++it;}cout << endl;//②范围for遍历for (auto& kv : dict){cout << kv.first << ":" << kv.second << endl;}cout << endl;

结果如图:

由打印结果可知:

map的遍历顺序并不是按照插入顺序遍历的,而是按照ASCII码表的顺序遍历的;

map与set相似,插入的key相同,value不同,不会插入也不会更新。

🥉计数+[ ]的使用

 ①:

//①string arr[] = { "苹果", "西瓜", "苹果", "西瓜", "苹果", "苹果", "西瓜",
"苹果", "香蕉", "苹果", "西瓜", "香蕉", "草莓" };map<string, int> countMap;for (auto& e : arr){map<string, int>::iterator it = countMap.find(e);if (it != countMap.end()){it->second++;}else{countMap.insert(make_pair(e, 1));}}for (auto& kv : countMap){cout << kv.first << ":" << kv.second << endl;}cout << endl;

结果如图:

②:

operator[]的原理是:  用<key,T()>构造一个键值对,然后调用insert()函数将该键值对插入到map中  ,如果key已经存在,插入失败,insert函数返回该key所在位置的迭代器  

如果key不存在,插入成功,insert函数返回新插入元素所在位置的迭代器,operator[]函数最后将insert返回值键值对中的value返回

详解会体现在代码里:

//②string arr[] = { "苹果", "西瓜", "苹果", "西瓜", "苹果", "苹果", "西瓜",
"苹果", "香蕉", "苹果", "西瓜", "香蕉", "草莓" };map<string, int> countMap;for (auto& e : arr){pair<map<string, int>::iterator, bool> ret;ret = countMap.insert(make_pair(e, 1));//insert的返回值类型为pair//而insert的返回值类型pair的第一个参数为iterator,第二个参数为bool//若“e”已经存在,则插入失败,insert返回该“e”所在位置的迭代器和false//若“e”不存在,则插入成功,insert返回新插入元素所在位置的迭代器和true//而insert的返回值-迭代器解引用得到的也是pair类型,此pair类型的第一个参数为key,第二个参数为value//也可以理解为:insert的返回值为(kv,bool),kv的返回值为(k,v)if (ret.second == false)//ret的first是kv,second是bool{ret.first->second++;//first是kv,second是v}}for (auto& kv : countMap){cout << kv.first << ":" << kv.second << endl;}cout << endl;

 上一段代码简单介绍map的 [ ] 是如何实现的:

V& operator[](const K& key)
{pair<iterator, bool> ret = insert(make_pair(key, V()));return ret.first->second;
}

那么②的代码就可以简写为:

//②--->③string arr[] = { "苹果", "西瓜", "苹果", "西瓜", "苹果", "苹果", "西瓜",
"苹果", "香蕉", "苹果", "西瓜", "香蕉", "草莓" };map<string, int> countMap;for (auto& e : arr){countMap[e]++;}for (auto& kv : countMap){cout << kv.first << ":" << kv.second << endl;}cout << endl;

 结果如图:

[]总结: 

 所以[]又兼具:

①插入;②查找;③修改;④插入+修改

int main()
{map<string, string> dict;dict.insert(make_pair("apple", "苹果"));dict.insert(make_pair("pear", "梨"));//插入dict["插入"];//查找cout << dict["pear"] << endl;//修改dict["apple"] = "大苹果";//插入+修改dict["pear2"] = "梨2";for (auto& kv : dict){cout << kv.first << ":" << kv.second << endl;}return 0;
}

🥈map小结

①map中的的元素是键值对;

②map中的key是唯一的,并且不能修改;

③默认按照小于的方式对key进行比较;

④map中的元素如果用迭代器去遍历,可以得到一个有序的序列;

⑤map的底层为平衡搜索树(红黑树),查找效率比较高O(log_2 N);

⑥支持[]操作符,operator[]中实际进行插入查找。

🥇multimap

🥈multimap的官网介绍

1. Multimaps是关联式容器,它按照特定的顺序,存储由key和value映射成的键值对,其中多个键值对之间的key是可以重复的;

2. 在multimap中,通常按照key排序和惟一地标识元素,而映射的value存储与key关联的内容。key和value的类型可能不同,通过multimap内部的成员类型value_type组合在一起, value_type是组合key和value的键值对: typedef pair value_type;

3. 在内部,multimap中的元素总是通过其内部比较对象,按照指定的特定严格弱排序标准对 key进行排序的;

4. multimap通过key访问单个元素的速度通常比unordered_multimap容器慢,但是使用迭代 器直接遍历multimap中的元素可以得到关于key有序的序列;

5. multimap在底层用二叉搜索树(红黑树)来实现。

🥈multimap的使用

 multimap与map类似,不同点是,multimap支持冗余:

    multimap<string, string> dict;dict.insert(make_pair("banana", "香蕉"));dict.insert(make_pair("banana", "香蕉2"));dict.insert(make_pair("peach", "桃子"));dict.insert(make_pair("peach", "桃子2"));dict.insert(make_pair("pear", "梨"));dict.insert(make_pair("apple", "苹果"));for (auto& kv : dict){cout << kv.first << ":" << kv.second << endl;}cout << endl;

结果如图:

🥈multimap小结

①multimap中的key是可以重复的;

② multimap中的元素默认将key按照小于来比较;

③multimap中没有重载operator[]操作(同学们可思考下为什么?);

④使用时与map包含的头文件相同。

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

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

相关文章

浅析云场景SSD实时迁移技术

在数据中心的运营管理中&#xff0c;负载均衡和系统容错是确保高效稳定运行的关键。SSD实时迁移技术&#xff0c;为解决这些问题提供了创新方案&#xff0c;成为数据中心技术发展的重要驱动力。 以AI训练任务为例&#xff0c;其运行时间长且无需用户频繁交互。数据中心的负载会…

同一局域网远程控制其他电脑以及Windows家庭版开启远程桌面等解决方法

1. 前言 家庭版的 Windows 10 和 Windows 11 默认不支持远程桌面功能。然而&#xff0c;我们可以通过使用 RDPWrap 项目来启用这一功能。 电脑的“设置”-> “远程桌面设置”中查看系统是否支持远程桌面 2.下载安装 RDPWrap 安装该项目使家庭版也支持远程桌面 项目地址&…

DeepSeek-R1:将强化学习用于激励大型语言模型的推理能力

目录 引言 一、DeepSeek-R1的贡献 二、DeepSeek-R1的方法 2.1、DeepSeek-R1-Zero&#xff1a;基础模型上的强化学习 2.2、DeepSeek-R1&#xff1a;冷启动强化学习 2.3、蒸馏&#xff1a;赋予小模型推理能力 三、DeepSeek-R1实验结果 3.1、模型优点 3.2、模型缺点 四、…

数据库管理-第287期 Oracle DB 23.7新特性一览(20250124)

数据库管理287期 20245-01-24 数据库管理-第287期 Oracle DB 23.7新特性一览&#xff08;20250124&#xff09;1 AI向量搜索&#xff1a;算术和聚合运算2 更改Compatible至23.6.0&#xff0c;以使用23.6或更高版本中的新AI向量搜索功能3 Cloud Developer包4 DBMS_DEVELOPER.GET…

Android BitmapShader简洁实现马赛克,Kotlin(二)

Android BitmapShader简洁实现马赛克&#xff0c;Kotlin&#xff08;二&#xff09; 这一篇 Android BitmapShader简洁实现马赛克&#xff0c;Kotlin&#xff08;一&#xff09;-CSDN博客 遗留一个问题&#xff0c;xml定义的MyView为wrap_content的宽高&#xff0c;如果改成其…

智能化加速标准和协议的更新并推动验证IP(VIP)在芯片设计中的更广泛应用

作者&#xff1a;Karthik Gopal, SmartDV Technologies亚洲区总经理 智权半导体科技&#xff08;厦门&#xff09;有限公司总经理 随着AI技术向边缘和端侧设备广泛渗透&#xff0c;芯片设计师不仅需要考虑在其设计中引入加速器&#xff0c;也在考虑采用速度更快和带宽更高的总…

【问题】Chrome安装不受支持的扩展 解决方案

此扩展程序已停用&#xff0c;因为它已不再受支持 Chromium 建议您移除它。详细了解受支持的扩展程序 此扩展程序已停用&#xff0c;因为它已不再受支持 详情移除 解决 1. 解压扩展 2.打开manifest.json 3.修改版本 将 manifest_version 改为3及以上 {"manifest_ver…

Vue入门(Vue基本语法、axios、组件、事件分发)

Vue入门 Vue概述 Vue (读音/vju/&#xff0c;类似于view)是一套用于构建用户界面的渐进式框架&#xff0c;发布于2014年2月。与其它大型框架不同的是&#xff0c;Vue被设计为可以自底向上逐层应用。Vue的核心库只关注视图层&#xff0c;不仅易于上手&#xff0c;还便于与第三…

[云讷科技]Kerloud Falcon四旋翼飞车虚拟仿真空间发布

虚拟仿真环境作为一个独立的专有软件包提供给我们的客户&#xff0c;用于帮助用户在实际测试之前验证自身的代码&#xff0c;并通过在仿真引擎中添加新的场景来探索新的飞行驾驶功能。 环境要求 由于环境依赖关系&#xff0c;虚拟仿真只能运行在装有Ubuntu 18.04的Intel-64位…

postgresql15的启动

PostgreSQL是一个功能非常强大的、源代码开放的客户/服务器关系型数据库管理系统&#xff0c;且因为许可证的灵活&#xff0c;任何人都可以以任何目的免费使用、修改和分发PostgreSQL。现在国产数据库大力发展阶段&#xff0c;学习和熟悉postgresql的功能是非常有必要的&#x…

Python NumPy(1):介绍、Ndarray对象、NumPy数据类型

1 介绍 NumPy(Numerical Python) 是 Python 语言的一个扩展程序库&#xff0c;支持大量的维度数组与矩阵运算&#xff0c;此外也针对数组运算提供大量的数学函数库。NumPy 是一个运行速度非常快的数学库&#xff0c;主要用于数组计算&#xff0c;包含&#xff1a; 一个强大的N…

sql语句--新手入门增删改查保姆级教学

一丶在数据库管理系统中&#xff0c;SQL&#xff08;Structured Query Language&#xff0c;结构化查询语言&#xff09;是用于访问和操作数据库的标准编程语言。以下将通过一个假设的“articles”表&#xff08;代表文章或博客条目&#xff09;来展示SQL中的增&#xff08;Ins…

INCOSE需求编写指南-第1部分:介绍

第1部分&#xff1a;介绍Section 1: Introduction 1.1 目的和范围 Purpose and Scope 本指南专门介绍如何在系统工程背景下以文本形式表达需求和要求陈述。其目的是将现有标准&#xff08;如 ISO/IEC/IEEE 29148&#xff09;中的建议以及作者、主要贡献者和审稿员的最佳实践结…

Windows上通过Git Bash激活Anaconda

在Windows上配置完Anaconda后&#xff0c;普遍通过Anaconda Prompt激活虚拟环境并执行Python&#xff0c;如下图所示&#xff1a; 有时需要连续执行多个python脚本时&#xff0c;直接在Anaconda Prompt下可以通过在以下方式&#xff0c;即命令间通过&&连接&#xff0c;…

GIS 中的 SQLAlchemy:空间数据与数据库之间的桥梁

利用 SQLAlchemy 在现代应用程序中无缝集成地理空间数据导言 地理信息系统&#xff08;GIS&#xff09;在管理城市规划、环境监测和导航系统等各种应用的空间数据方面发挥着至关重要的作用。虽然 PostGIS 或 SpatiaLite 等专业地理空间数据库在处理空间数据方面非常出色&#…

MySQL中的读锁与写锁:概念与作用深度剖析

MySQL中的读锁与写锁&#xff1a;概念与作用深度剖析 在MySQL数据库的并发控制机制中&#xff0c;读锁和写锁起着至关重要的作用。它们是确保数据在多用户环境下能够正确、安全地被访问和修改的关键工具。 一、读锁&#xff08;共享锁&#xff09;概念 读锁&#xff0c;也称为…

SpringBoot 实现动态管理定时任务 Job的动态操作(添加、修改、启停、执行、删除)以及界面展示和具体Job的创建与执行示例

SpringBoot 实现动态管理定时任务 Job的动态操作&#xff08;添加、修改、启停、执行、删除&#xff09;以及界面展示和具体Job的创建与执行示例 关键接口类&#xff1a; CronTaskRegistrar SchedulingRunnable . 添加定时任务注册类&#xff0c;用来增加、删除定时任务 impo…

LabVIEW太赫兹二维扫描成像系统

使用LabVIEW设计太赫兹二维扫描成像系统。通过LabVIEW平台开发&#xff0c;结合硬件如太赫兹源、平移台、锁相放大器等&#xff0c;实现了高效、精准的成像功能。系统采用蛇形扫描方式&#xff0c;通过动态调整扫描参数&#xff0c;达到优化成像质量的目的。 ​ 项目背景 在非…

Spring 核心技术解析【纯干货版】- V:Spring 基础模块 Spring-Context 模块精讲

Spring 框架作为 Java 开发领域最流行的框架之一&#xff0c;其核心模块承载了大量企业级应用开发的基础功能。在 Spring 的核心模块中&#xff0c;Spring-Context 模块尤为重要&#xff0c;它不仅提供了应用上下文的管理功能&#xff0c;还扩展了事件驱动、国际化支持、资源加…

2025年国产化推进.NET跨平台应用框架推荐

2025年国产化推进.NET跨平台应用框架推荐 1. .NET MAUI NET MAUI是一个开源、免费&#xff08;MIT License&#xff09;的跨平台框架&#xff08;支持Android、iOS、macOS 和 Windows多平台运行&#xff09;&#xff0c;是 Xamarin.Forms 的进化版&#xff0c;从移动场景扩展到…