【C++干货铺】STL中set和map的介绍和使用

=========================================================================

个人主页点击直达:小白不是程序媛

C++系列专栏:C++干货铺

代码仓库:Gitee

=========================================================================

目录

序列式容器

关联式容器

键值对

树形结构的关联式容器

set

set的介绍

set的使用

set的模板参数列表

set的构造

​编辑 set的容量

set的删除和查找

multiset

multiset的介绍

multiset的使用

map

map的介绍

map的使用

map的模板参数说明

map的迭代器

map的容量与元素的访问

map中元素的修改

map的终极操作operator[] 

multimap 

multimap的介绍

multimap的使用


序列式容器

在之前的文章中,我们已经接触过STL中的部分容器,比如:vector、list、deque、forward_list(C++11)等,这些容器统称为序列式容器,因为其底层为线性序列数据结构,里面存储的是元素本身。


关联式容器

关联式容器也是用来存储数据的,与序列式容器不同的是,其里面存储的是<key, value>结构键值对,在数据检索时比序列式容器效率更高


键值对

用来表示具有一一对应关系的一种结构,该结构中一般只包含两个成员变量key和value,key代
表键值,value表示与key对应的信息。
比如:现在要建立一个英汉互译的字典,那该字典中必然
有英文单词与其对应的中文含义,而且,英文单词与其中文含义是一一对应的关系,即通过该应
该单词,在词典中就可以找到与其对应的中文含义。

SGI—STL中对于键值的定义:

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){}
};

树形结构的关联式容器

根据应用场景的不桶,STL总共实现了两种不同结构的管理式容器:树型结构哈希结构

树型结构的关联式容器主要有四种:map、set、multimap、multiset

这四种容器的共同点是:使用平衡搜索树(即红黑树)作为其底层结果,容器中的元素是一个有序的序列。下面一依次介绍每一个容器。 


set

set的介绍

set的文档介绍

翻译:
1. set是按照一定次序存储元素的容器
2. 在set中,元素的value也标识它(value就是key,类型为T),并且每个value必须是唯一的。
set中的元素不能在容器中修改(元素总是const),但是可以从容器中插入或删除它们。
3. 在内部,set中的元素总是按照其内部比较对象(类型比较)所指示的特定严格弱排序准则进行排序。
4. set容器通过key访问单个元素的速度通常比unordered_set容器慢,但它们允许根据顺序对
子集进行直接迭代。
5. set在底层是用二叉搜索树(红黑树)实现的。 

注意:
1. 与map/multimap不同,map/multimap中存储的是真正的键值对<key, value>,set中只放
value,但在底层实际存放的是由<value, value>构成的键值对。
2. set中插入元素时,只需要插入value即可,不需要构造键值对。
3. set中的元素不可以重复(因此可以使用set进行去重)。
4. 使用set的迭代器遍历set中的元素,可以得到有序序列
5. set中的元素默认按照小于来比较
6. set中查找某个元素,时间复杂度为:log2(n)
7. set中的元素不允许修改
8. set中的底层使用二叉搜索树(红黑树)来实现。

set的使用

set的模板参数列表

T: set中存放元素的类型,实际在底层存储<value, value>的键值对。
Compare:set中元素默认按照小于来比较
Alloc:set中元素空间的管理方式,使用STL提供的空间配置器管理

set的构造

函数声明功能介绍
set(const Compare&comp = Compare(), const Allocator& = Allocator() );构造空set

set(Inputlterator first , Inputlterator last , const Com pare& comp = Compare

,cosnst Allocator& = Allocator() );

用[first,last)区间中的元素构造set
set (const set<Key,Compare, Allocator>&x);set的拷贝构造
void test_set()
{//空构造set<int> s1;s1.insert(4);s1.insert(3);s1.insert(2);s1.insert(1);//拷贝构造set<int> s2 = s1;//区间构造int arr[] = { 1,2,3,4 };set<int> s3(arr, arr + 4);
}

set的迭代器

函数声明函数功能
iterator begin() 返回set中起始位置元素的迭代器
iterator end() 返回set中最后一个元素后面的迭代器
const_iterator cbegin()
const
返回set中起始位置元素的const迭代器
const_iterator cend() const 返回set中最后一个元素后面的const迭代器
reverse_iterator rbegin() 返回set第一个元素的反向迭代器,即end
reverse_iterator rend()返回set最后一个元素下一个位置的反向迭代器,
即rbegin
const_reverse_iterator
crbegin() const
返回set第一个元素的反向const迭代器,即cend
const_reverse_iterator
crend() const
返回set最后一个元素下一个位置的反向const迭
代器,即crbegin
void test_set1()
{set<int> s;s.insert(4);s.insert(4);s.insert(3);s.insert(3);s.insert(2);pair<set<int>::iterator,bool> pa = s.insert(2);cout << pa.second << endl;s.insert(1);auto pi = s.insert(1);cout << pi.second << endl;//迭代器set<int>::iterator it = s.begin();while (it != s.end()){cout << *it << " ";it++;}cout << endl;for (auto& e : s){cout << e << " ";}cout << endl;
}

 set的容量

函数声明函数功能
bool empty () const检测set是否为空,空返回true,否则返回false;
size_type size() cosnt返回set中的有效元素的个数;
void test_set2()
{set<int> s;s.insert(4);s.insert(4);s.insert(2);s.insert(3);cout << s.empty() << endl;cout << s.size() << endl;
}

set的删除和查找

函数声明函数功能
void erase(iterator first position)删除set中position位置上的元素
size_type erase (const key_type& x)删除set中值为x的元素,返回删除元素的个数
void earse(iterator first , iterator last)删除set中[first,last)区间中的元素
iterator find(const key_type& x)const返回set中值为x的元素的位置
size_type count(const key_type& x) const返回set中值为x的元素个数
void test_set3()
{set<int>s;s.insert(6);s.insert(5);s.insert(4);s.insert(3);s.insert(2);s.insert(1);//直接删除s.erase(4);for (auto& e : s){cout << e << " ";}cout << endl;//查找set<int>::iterator it = s.find(2);//不进行判断删除end(),程序会崩溃;s.erase(20);//删除不存在的值倒没有什么影响if (it != s.end()){s.erase(3);}for (auto& e : s){cout << e << " ";}cout << endl;cout << s.count(2) << endl;
}

multiset

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底层结构为二叉搜索树(红黑树)

注意:
1. multiset中再底层中存储的是<value, value>的键值对
2. mtltiset的插入接口中只需要插入即可
3. 与set的区别是,multiset中的元素可以重复,set是中value是唯一的
4. 使用迭代器对multiset中的元素进行遍历,可以得到有序的序列                                            5. multiset中的元素不能修改
6. 在multiset中找某个元素,时间复杂度为O(log2(n))
7. multiset的作用:可以对元素进行排序

multiset的使用

multiset和set的使用接口都是相同的,只有一些函数返回值的使用不同。这里只演示一些不同的地方。

void test_multiset()
{int arr[] = { 4,3,2,1,4,3,2,1 };multiset<int> s(arr, arr + 8);for (auto& e : s){cout << e << " ";}cout << endl;//返回的为第一个出现的值multiset<int>::iterator it = s.find(2);while (it != s.end()){cout << *it << " ";it++;}cout << endl;//值为x的元素有几个cout << s.count(1) << endl;//删除了几个值为x的元素size_t n = s.erase(1);cout << n << endl;//equal_range 返回和x值相等的第一个位置和最后一位置pair<multiset<int>::iterator, multiset<int>::iterator> pi = s.equal_range(2);//使用迭代器区间删除元素s.erase(pi.first, pi.second);for (auto& e: s){cout << e << " ";}}

map

map的介绍

map的文档介绍

翻译:
1. map是关联容器,它按照特定的次序(按照key来比较)存储由键值key和值value组合而成的元素。
2. 在map中,键值key通常用于排序和惟一地标识元素,而值value中存储与此键值key关联的
内容。键值key和值value的类型可能不同,并且在map的内部,key与value通过成员类型
value_type绑定在一起,为其取别名称为pair:typedef pair<const key, T> value_type;
3. 在内部,map中的元素总是按照键值key进行比较排序的。
4. map中通过键值访问单个元素的速度通常比unordered_map容器慢,但map允许根据顺序
对元素进行直接迭代(即对map中的元素进行迭代时,可以得到一个有序的序列)。
5. map支持下标访问符,即在[]中放入key,就可以找到与key对应的value。
6. map通常被实现为二叉搜索树(更准确的说:平衡二叉搜索树(红黑树))。 

map的使用

map的模板参数说明

key: 键值对中key的类型
T: 键值对中value的类型
Compare: 比较器的类型,map中的元素是按照key来比较的,缺省情况下按照小于来比
较,一般情况下(内置类型元素)该参数不需要传递,如果无法比较时(自定义类型),需要用户自己显式传递比较规则(一般情况下按照函数指针或者仿函数来传递)
Alloc:通过空间配置器来申请底层空间,不需要用户传递,除非用户不想使用标准库提供的空间配置器。

map的构造

函数声明功能介绍
map()构造一个空的map
void test_map()
{//空构造map<string, string> dict;//需要使用键对值进行初始化dict.insert(pair<string, string>("sort", "排序"));dict.insert(pair<string, string>("insert", "插入"));dict.insert(pair<const char*, const char*>("left", "左边"));//make_pair函数的返回值为一个键对值dict.insert(make_pair("right", "右边"));string s1("key"), s2("value");dict.insert(make_pair(s1,s2));
}

map的迭代器

函数声明

功能简介
begin()和end()                                                      begin:首元素位置,end:最后一个元素位置
cbegin和cend()与begin和end的意义相同,但是所指向的元素不可以修改
rbegin和rend()反向迭代器
crbegin()和crend()反向迭代器向的位置不可以修改
void test_map1()
{map<string, string> dict;//需要使用键对值进行初始化dict.insert(pair<string, string>("sort", "排序"));dict.insert(pair<string, string>("insert", "插入"));dict.insert(pair<const char*, const char*>("left", "左边"));//make_pair函数的返回值为一个键对值dict.insert(make_pair("right", "右边"));string s1("key"), s2("value");dict.insert(make_pair(s1, s2));//迭代器map<string, string>::iterator it = dict.begin();while (it != dict.end()){//map键对值中的第一个数据是不支持修改的,第二个可以修改。//it->first++;it->second += 'x';cout << (*it).first << ":" << (*it).second << endl;it++;}cout << endl;//支持迭代器一定支持范围forfor (auto& e : dict){cout << e.first << ":" << e.second << endl;}
}

map的容量与元素的访问

函数声明功能简介
bool empty() const检测map中的元素是否为空,是返回true,否则返回false
size_type size() const返回map中的有效元素个数
mapped_type& operator[] (const key_type& k)返回key对应的value
void test_map2()
{map<string,string> s;//判断是否为空,空返回1非空返回0cout << s.empty() << endl;//返回有效元素个数s.insert(make_pair("xxx", "yyy"));cout << s.size() << endl;//返回key对应的valuecout << s.operator[]("xxx") << endl;cout << s.operator[]("yyy") << endl;
}

当key不在map中时,通过operator[]获取对应的value会发生什么问题?

注意:在元素访问时,有一个与operator[]类似的操作at()(该函数不常用)函数,都是通过
key找到与key对应的value然后返回其引用,不同的是:当key不存在时,operator[]用默认
value与key构造键值对然后插入,返回该默认value,at()函数直接抛异常。

map中元素的修改

函数声明功能简介
void erase ( iterator position ) 删除position位置上的元素
size_type erase ( const key_type& x )删除键值为x的元素
iterator find ( const key_type& x)在map中插入key为x的元素,找到返回该元
素的位置的迭代器,否则返回end
void test_map4()
{map<string, string> m;m.insert(make_pair("sort", "排序"));m.insert(make_pair("set", "组"));m.insert(make_pair("map", "图"));m.insert(make_pair("rigth", "右边"));m.insert(make_pair("left", "左边"));for (auto& e : m){cout << e.first << ":" << e.second << endl;}cout << endl;m.erase("sort");map<string, string>::iterator it = m.find("set");//删除end位置的值会崩溃if (it != m.end()){m.erase(it);}for (auto& e : m){cout << e.first << ":" << e.second << endl;}cout << endl;
}

map的终极操作operator[] 

map::operator[]文档介绍

map中的operator[]函数可以实现插入、查找、修改等一系列功能,该函数的实现也特别复杂;

void test_map3()
{//operator[] 插入 查找 修改map<string, string> m;//插入m["insert"] = "插入";m["left"] = "左边";//查找cout << m["insert"] << endl;cout << m["left"]  <<endl;//修改m["left"] = "剩余";cout << m["left"] << endl;
}

【map总结】
1. map中的的元素是键值对
2. map中的key是唯一的,并且不能修改
3. 默认按照小于的方式对key进行比较
4. map中的元素如果用迭代器去遍历,可以得到一个有序的序列
5. map的底层为平衡搜索树(红黑树),查找效率比较高$O(log_2 N)$
6. 支持[]操作符,operator[]中实际进行插入查找。 


multimap 

multimap的介绍

multimap的文档介绍

翻译:
1. Multimaps是关联式容器,它按照特定的顺序,存储由key和value映射成的键值对<key,
value>,其中多个键值对之间的key是可以重复的。
2. 在multimap中,通常按照key排序和惟一地标识元素,而映射的value存储与key关联的内
容。key和value的类型可能不同,通过multimap内部的成员类型value_type组合在一起,
value_type是组合key和value的键值对:
typedef pair<const Key, T> value_type;
3. 在内部,multimap中的元素总是通过其内部比较对象,按照指定的特定严格弱排序标准对
key进行排序的。
4. multimap通过key访问单个元素的速度通常比unordered_multimap容器慢,但是使用迭代
器直接遍历multimap中的元素可以得到关于key有序的序列。
5. multimap在底层用二叉搜索树(红黑树)来实现。

注意:multimap和map的唯一不同就是:map中的key是唯一的,而multimap中key是可以
重复的。

multimap的使用

multimap中的接口可以参考map,功能都是类似的。

注意:
1. multimap中的key是可以重复的。
2. multimap中的元素默认将key按照小于来比较
3. multimap中没有重载operator[]操作
4. 使用时与map包含的头文件相同


今天对set和map的介绍和使用到这里就结束了,希望大家读完后有很大的收获,也可以在评论区点评文章中的内容和分享自己的看法。您三连的支持就是我前进的动力,感谢大家的支持!! ! 

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

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

相关文章

HPM6750开发笔记《开发环境的搭建》

目录 一&#xff0c;下载完整的HPM—SDK 二&#xff0c;安装硬件驱动 二&#xff0c;软件激活 三&#xff0c;创建工程 1.用文档中给的方法创建工程&#xff1a; 2.用sdk_env_v1.3.0中提供的工具创建工程&#xff1a; 一&#xff0c;下载完整的HPM—SDK 下载网址&#x…

Jmeter 分布式压测

‍你可以使用 JMeter 来模拟高并发秒杀场景下的压力测试。这里有一个例子&#xff0c;它模拟了同时有 5000 个用户&#xff0c;循环 10 次的情况‍。 请求默认配置 token 配置 秒杀接口 结果分析 但是&#xff0c;实际企业中&#xff0c;这种压测方式根本不满足实际需求。下面介…

XPM_CDC_SINGLE(UG974)

Parameterized Macro: Single-bit Synchronizer&#xff08;参数化宏&#xff1a;单比特同步器&#xff09; MACRO_GROUP: XPMMACRO_SUBGROUP: XPM_CDCFamilies: UltraScale, UltraScale 1、 Introduction&#xff08;介绍&#xff09; 此宏将一个一位信号从源时钟域同步到目…

医院绩效考核系统源码,java源码,商业级医院绩效核算系统源码

医院绩效定义&#xff1a; “医院工作量绩效方案”是一套以工作量&#xff08;RBRVS&#xff0c;相对价值比率&#xff09;为核算基础&#xff0c;以工作岗位、技术含量、风险程度、服务数量等业绩为主要依据&#xff0c;以工作效率和效益、工作质量、患者满意度等指标为综合考…

Python 操作 MySQL:使用 mysql-connector-python 操作 MySQL 数据库

大家好&#xff0c;我是水滴~~ 当涉及到使用 Python 操作 MySQL 数据库时&#xff0c;mysql-connector-python 库是一个强大而常用的选择。该库提供了与 MySQL 数据库的交互功能&#xff0c;使您能够执行各种数据库操作&#xff0c;如连接数据库、执行查询和插入数据等。在本文…

Shell三剑客:awk(awk编辑编程)五

一、前言 AWK 可以使用关联数组这种数据结构&#xff0c;索引可以是数字或字符串。AWK关联数 组也不需要提前声明其大小&#xff0c;因为它在运行时可以自动的增大或减小。 二、数组语法格式 array_name[index]valuearray_name&#xff1a;数组的名称index&#xff1a;数组索…

【数据结构】C语言实现双链表的基本操作

双链表及其基本操作的实现 导言一、单链表与双链表二、双链表类型的创建三、双链表的初始化四、双链表的创建五、双链表的遍历六、双链表的查找七、双链表的插入八、双链表的删除结语 导言 大家好&#xff0c;很高兴又和大家见面啦&#xff01;&#xff01;&#xff01; 经过…

【adb】--- win10 配置 adb环境 超详细 (持续更新中)

在编程的艺术世界里&#xff0c;代码和灵感需要寻找到最佳的交融点&#xff0c;才能打造出令人为之惊叹的作品。而在这座秋知叶i博客的殿堂里&#xff0c;我们将共同追寻这种完美结合&#xff0c;为未来的世界留下属于我们的独特印记。 【adb】--- win10 配置 adb环境 超详细 &…

解决Qt“报无法定位程序输入点xxx于动态连接库“问题

今天&#xff0c;在使用QtVS2019编译工程时&#xff0c;弹出"无法定位程序输入点xxx于动态链接库"问题&#xff0c;如图(1)所示&#xff1a; 图(1) 报"无法定位程序输入点xxx于动态链接库"问题 出现这种问题的原因有很多&#xff1a; (1) 工程Release/Deb…

低代码选型注意事项

凭借着革命性的生产力优势&#xff0c;低代码技术火爆了整个IT圈。面对纷繁复杂的低代码和无代码产品&#xff0c;开发者该如何选择&#xff1f; 在研究低代码平台的年数上&#xff0c;本人已有3年&#xff0c;也算是个低代码资深用户了&#xff0c;很多企业面临低代码选型上的…

iOS 开发设计 App 上架符合要求的截图

1. 真机运行截屏 2. 可以在 Apple developer 官网 Design 下找到 iPhone 边框 https://developer.apple.com/design/resources/ 不用这个边框也行&#xff0c;可以参考已上架 App 的图片框 3. 使用 Procreate&#xff08;PhotoShop&#xff09;创建符合要求的画布大小 4. 导入…

听GPT 讲Rust源代码--src/tools(27)

File: rust/src/tools/clippy/clippy_lints/src/methods/suspicious_to_owned.rs 文件rust/src/tools/clippy/clippy_lints/src/methods/suspicious_to_owned.rs的作用是实施Clippy lint规则&#xff0c;检测产生潜在性能问题的字符转换代码&#xff0c;并给出相关建议。 在Rus…

智能优化算法应用:基于袋獾算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于袋獾算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于袋獾算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.袋獾算法4.实验参数设定5.算法结果6.参考文献7.MA…

【UE5.1】程序化生成Nanite植被

目录 效果 步骤 一、下载Gaea软件和树林资产 二、使用Gaea生成贴图 三、 生成地形 四、生成草地 五、生成树林 六、生成湖泊 七、其它功能介绍 7.1 调整树林生成的面积 7.2 让植物随风飘动 7.3 玩家和植物互动 7.4 雪中树林 7.5 环境音效 效果 步骤 一、下载Ga…

HBase 集群搭建

文章目录 安装前准备兼容性官方网址 集群搭建搭建 Hadoop 集群搭建 Zookeeper 集群解压缩安装配置文件高可用配置分发 HBase 文件 服务的启停启动顺序停止顺序 验证进程查看 Web 端页面 安装前准备 兼容性 1&#xff09;与 Zookeeper 的兼容性问题&#xff0c;越新越好&#…

信息泄露总结

文章目录 一、备份文件下载1.1 网站源码1.2 bak文件泄露1.3 vim缓存1.4 .DS_Store 二、Git泄露2.1 git知识点2.1 log2.2 stash 三、SVN泄露3.1 SVN简介3.2 SVN的文件3.3 SVN利用 四、Hg泄露 一、备份文件下载 1.1 网站源码 常见的网站源码备份文件后缀&#xff1a; tartar.gz…

非阻塞 IO(NIO)

文章目录 非阻塞 IO(NIO)模型驱动程序应用程序模块使用 非阻塞 IO(NIO) 上一节中 https://blog.csdn.net/tyustli/article/details/135140523&#xff0c;使用等待队列头实现了阻塞 IO 程序使用时&#xff0c;阻塞 IO 和非阻塞 IO 的区别在于文件打开的时候是否使用了 O_NONB…

深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第五节 引用类型复制问题及用克隆接口ICloneable修复

深入浅出图解C#堆与栈 C# Heaping VS Stacking 第五节 引用类型复制问题及用克隆接口ICloneable修复 [深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第一节 理解堆与栈](https://mp.csdn.net/mdeditor/101021023)[深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第二节…

spdlog中的异步日志方案

日志方案 同步日志方案&#xff1a;立即输出日志记录的方案才能继续执行其他任务。 异步日志方案&#xff1a;先抛出一个日志记录的任务到某个地方&#xff0c;不马上执行打印也不影响往下执行其他任务。 二者关键区别是产生日志记录并调用相关的日志任务接口之后&#xff0…

【Kafka】Kafka客户端认证失败:Cluster authorization failed.

背景 kafka客户端是公司内部基于spring-kafka封装的spring-boot版本&#xff1a;3.xspring-kafka版本&#xff1a;2.1.11.RELEASE集群认证方式&#xff1a;SASL_PLAINTEXT/SCRAM-SHA-512经过多年的经验&#xff0c;以及实际验证&#xff0c;配置是没问题的&#xff0c;但是业务…