【C++进阶】map和set的使用

news2025/1/20 1:09:54

【C++进阶】map和set的使用

🥕个人主页:开敲🍉

🔥所属专栏:C++🥭

🌼文章目录🌼

1. 序列式容器和关联式容器

2. set系列的使用

    2.1 set 和 multiset

    2.2 set 类的介绍

    2.3 set 的构造和迭代器

  2.4 set 的增删查

  2.5 multiset 和 set 的区别

    2.6 两道练习题

3. map 系列的使用

    3.1 map 和 multimap

    3.2 map 类的介绍

    3.3 pair 类型的介绍

    3.4 map 的构造和使用

    3.5 map 的增删查

    3.6 map 的数据修改

    3.7 构造遍历及增删查的用法参考

    3.8 multimap 和 map 的区别

    3.9 两道练习题

1. 序列式容器和关联式容器

  前面我们已经接触过STL中的部分容器如:string、vector、list、deque、array、forward_list等,这些容器统称为序列式容器,因为逻辑结构为线性的数据结构,两个相邻位置存储的值一般没有紧密的关联关系,比如交换或者修改,不会破坏它的结构。顺序容器中的元素是按它们的存储位置来顺序存储和访问的。

  关联式容器也是用来存储数据的,与序列式容器不同的是,关联式容器的逻辑结构通常是非线性结构的,两个位置有紧密的关联关系,比如将相邻两个位置交换,它的结构就被破坏了。关联式容器有 map/set 系列和 unordered_maxp/unordered_set 系列。

  本篇讲解的 map 和 set 的底层是 红黑树,也就是平衡二叉搜索树。set是 key 搜索场景的结构,map是 key/value (键和值)搜索场景的结构。

2. set系列的使用
    2.1 set 和 multiset

参考文档:<set> - C++ Reference (cplusplus.com)

    2.2 set 类的介绍

  set的声明如下:

T就是 set 底层关键字的类型。

  set 默认要求 T 支持大小的比较,如果不支持或者想要按照自己的需求比较,可以实现仿函数传给第二个模板参数。

  set 底层的存储数据的内存是从空间配置器中申请的,如果需要可以自己实现内存池,传给第三个参数。

  一般情况下,我们不需要传后两个模板参数,用它的缺省参数即可。

  set 的底层是红黑树,增删查的效率是 O(logN),迭代器遍历时走的是二叉搜索树的中虚遍历,因此遍历结果是有序的,默认为升序。

  前面部分我们已经学习了 vector / list 等容器的使用,STL容器的接口设计高度相似,因此不需要再一个个的去看是什么功能,这里重点讲几个比较重要的:

insert:往 set 中插入元素,插入的方式和二叉搜索树一致,插入值比当前节点小走左边,比当前节点大走右边。

支持指定值插入、迭代器插入。

用法:

不支持插入相同的值:

erase:删除 set 中的节点,删除方式和二叉搜索树一致:如果是叶子节点直接删除,非叶子节点分为两种情况:① 有一个孩子,将孩子托付给父亲,删除节点  ② 有两个孩子,去到右子树中找最小值所在节点,将要删除节点的值和最小值节点的值交换,随后删除最小值节点。

支持指定元素删除、迭代器删除。

用法:

lower_bound:指定一个值,返回 set 中第一个 ≥ 该值的迭代器

用法:

upper_bound:指定一个值,返回 set 中第一个 > 该值的迭代器

find:寻找 set 中指定元素。

用法:

count:返回 set 中指定元素的个数,但是由于 set 不允许重复元素插入,因此这里返回的只有 0 和 1 ,但是这里还是返回个数是因为要和后面的 multiset 保持一致。

用法:

    2.3 set 的构造和迭代器

  set 的构造我们关注以下几个接口即可:

无参默认构造:

迭代器区间构造:

拷贝构造:

④ initializer list:initializer 列表构造:

  set 支持正向和反向迭代遍历,遍历默认按升序排序,因为底层是二叉搜索树,遍历时按照中序遍历;支持迭代器就意味着支持范围 for,在上面接口的用法中我也用到了范围 for。set 的迭代器不支持迭代器修改数据,如果修改关键字数据,会破坏 set 的结构。

  2.4 set 的增删查

  set 的增删查我们关注以下接口即可:

增:insert

单个数据插入。如果数据已经存在于 set 中,则插入失败

列表插入。同样的,set 中已经存在的值不会插入

迭代器区间插入。重复的值不会插入

删:erase

删除迭代器位置的值

删除指定元素。如果不存在则删除失败,返回00;反之,返回1

 删除一段迭代器区间的值

查:find、count、lower_bound、upper_bound

查找某一个值,返回该值所在的迭代器;如果找不到则返回 end()迭代器

查找某一个值,返回该值的个数;如果找不到,则该值的个数为0

 返回第一个大于等于 指定元素 位置的迭代器

返回第一个大于 指定元素 位置的点带起

  2.5 multiset 和 set 的区别

  multiset 和 set 的使用完全类似,主要区别在于 multiset 支持重复数据的存储,那么因此 insert、erase、find、count 都根据这种特性进行了改变,这里直接看用法的区别:

① insert:

可以看到,这里可以插入相同的值。

② erase:

可以看到,当我们删除 4 时,会将 ms 中所有的 4 都删掉。

③ find:

查找时,如果有多个相同的元素比如 7,则会返回 中序遍历 的第一个 7 。

④ count:

返回 multiset 中该元素的个数。

    2.6 两道练习题

349. 两个数组的交集 - 力扣(LeetCode)

LCR 022. 环形链表 II - 力扣(LeetCode)

  不论你之前使用什么方法做的,如果没有用 set 来做过,强烈建议你试试,完全就是降维打击。

3. map 系列的使用
    3.1 map 和 multimap

 参考文档:<map> - C++ Reference (cplusplus.com)

    3.2 map 类的介绍

  map 的声明如下,key 就是关键字,T 是 map底层 value 的类型,set 默认要求 key 支持大小比较,如果不支持或者想要按照自己的需要进行比较,则需要自行实现仿函数传给第三个模板参数,map 底层存储数据的内存是从空间配置器申请的。一般情况下,我们不需要传后两个模板参数。map 的底层是红黑树,增删查改的效率是 O(logN),迭代器遍历走的是二叉搜索树的中序遍历,因此遍历结果是有序的。

    3.3 pair 类型的介绍

  map 底层的红黑树节点中的数据,使用 pair<key,T> 来存储键值对数据。下面是库中的 pair,可以参考:

    3.4 map 的构造和使用

  map 的构造和使用我们关注以下几个接口即可(用法和 set 完全类似,这里就不演示用法了):

无参数默认构造:

迭代器区间构造:

拷贝构造:

列表构造:

insert:与 set 一样,都是按照二叉搜索树的结构插入数据。区别在于,map 支持重复数据的存储。

erase:指定元素删除。跟 set 的区别在于,如果有多个重复的指定元素,map 会全部删除。

find:查找指定元素。和 set 一样,返回指定元素所在的迭代器。

count:查找指定元素的个数。和 set 一样,返回指定元素的个数。

lower_bound:返回第一个 ≥ 指定元素的迭代器。

upper_bound:返回第一个 > 指定元素的迭代器。

operator[]:因为 map 支持修改,因此重载了 [] 符号,使得能够修改 map 中的值value,但是不能修改 key

  map 支持正向和反向迭代器遍历,遍历默认按照键 key 升序顺序,因为底层是二叉搜索树,遍历走的是中序遍历;支持迭代器就意味着支持范围 for,map 支持修改 value,不支持修改 key,因为修改关键字会破坏结构。

    3.5 map 的增删查

  map 的增删查关注以下几个接口:

增:

指定元素插入:

迭代器位置插入:

迭代器区间插入:

列表插入:

删:

迭代器位置删除:

指定元素删除:

迭代器区间删除:

查找:find、count、lower_bound、upper_bound

查找指定元素,返回指定元素所在迭代器:

查找指定元素,并返回指定元素的个数:

查找第一个 ≥ 指定元素的迭代器并返回:

查找第一个 > 指定元素的迭代器并返回:

    3.6 map 的数据修改

  前面提高了 map 支持修改 value,而不支持修改 key,那么 map 如何修改 value 呢?

  第一个方法就是通过迭代器修改,使用迭代器遍历时或者使用 find 查找返回时,修改迭代器所在key 的 value。

  第二个方法上面也有提及,就是 operator[],map 重载了 [],就是方便我们进行修改操作。当然,operator[],不仅仅可以进行修改操作,还可以进行插入和查找数据的操作,因此它是一个多功能接口。

用法:

    3.7 构造遍历及增删查的用法参考

① 构造+遍历:

② insert:

③ erase:erase 与 set 完全类似,都是根据键 key 去删除节点,map 无非就是多删除了一个 value,这里就不演示了。

    3.8 multimap 和 map 的区别

  multimap 和 map 的使用完全类似,主要区别点在于 multimap 支持关键值 key 重复,因此 insert、erase、count、find 都会因为这个特性与 map 有所差异。这里跟 set 和 multiset 的区别是一样的,比如 find 的元素有多个,则返回中序遍历的第一个。最后就是 multimap 不支持[],也就是没有重载 operator[],因为支持重复的键 key,因此只能插入而不能修改。

    3.9 两道练习题

138. 随机链表的复制 - 力扣(LeetCode)

692. 前K个高频单词 - 力扣(LeetCode)

                                               创作不易,点个赞呗,蟹蟹啦~

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

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

相关文章

【Linux篇】常用命令及操作技巧(基础篇)

&#x1f30f;个人博客主页&#xff1a;意疏-CSDN博客 希望文章能够给到初学的你一些启发&#xff5e; 如果觉得文章对你有帮助的话&#xff0c;点赞 关注 收藏支持一下笔者吧&#xff5e; 阅读指南&#xff1a; 开篇说明帮助命令常见的七个linux操作终端实用的技巧跟文件目录…

C++11之统一的列表初始化

一.{}初始化 在c98中&#xff0c;标准允许使用{}对数组或结构体元素进行统一的列表初始值设定&#xff1a; struct mess {int _x;string _str; }; int main() {//注意&#xff0c;使用new的一定是指针int* arr new int[4] {1, 2, 3, 4};//数组初始化int arr[] { 1,3,5,6 };…

基于Spring Boot和Vue的私人牙科诊所系统的设计与实现(毕业论文)

目 录 1 前言 1 1.1 研究目的与意义 1 1.2 国内外研究概况 1 1.3 主要研究内容 2 1.4 论文结构 3 2 系统分析 3 2.1 可行性分析 3 2.1.1 技术可行性分析 3 2.1.2 经济可行性分析 3 2.1.3 操作可行性分析 4 2.1.4 法律可行性分析 4 2.2 需求分析 4 2.2.1 管理员需求分析 4 2.2.2…

3.1 数据表的基本查询

我们学习的怎么管理逻辑空间&#xff0c;怎么创建数据表&#xff0c;怎么定义字段&#xff0c;怎么创建索引&#xff0c;这些都是DDL语句。从本次课开始&#xff0c;我们来学习DML语句&#xff0c;也就是该如何增删改查操作数据。我们学习DML语句的前提是数据表要有足够多的数据…

Moving Elevator System Fully functional

这是一个功能齐全的电梯系统,配有电梯箱车、电梯井、电缆和每层的门框 电梯完全被操纵,有动画门、电缆线、滑轮系统。 还有几个C#脚本文件控制电梯、门和灯。 此套餐还包括相关声音,如电梯移动、门打开/关闭、楼层铃叮。 电梯车厢有工作门和按钮,车顶还有一个逃生舱口。 每…

低投入、高效率 基于PHP+MySQL组合开发的求职招聘小程序源码系统 带完整的安装代码包以及搭建部署教程

系统概述 这款求职招聘小程序源码系统是专门为求职招聘领域打造的综合性平台。它利用 PHP 强大的编程语言特性和 MySQL 稳定的数据存储功能&#xff0c;实现了一个功能齐全、性能优越的求职招聘系统。 整个系统架构设计合理&#xff0c;具备良好的扩展性和兼容性。无论是小型…

从《中国数据库前世今生》看中国数据库技术的发展与挑战

从《中国数据库前世今生》看中国数据库技术的发展与挑战 引言 在当今数字化浪潮中&#xff0c;数据库技术已成为支撑全球经济运行的核心基础设施。作为程序员&#xff0c;我一直对数据库技术的发展充满好奇。《中国数据库前世今生》纪录片深入探索了中国数据库技术的演变历程…

【Python报错已解决】libpng warning: iccp: known incorrect sRGB profile

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 专栏介绍 在软件开发和日常使用中&#xff0c;BUG是不可避免的。本专栏致力于为广大开发者和技术爱好者提供一个关于BUG解决的经…

怎么把图片压缩变小?把图片压缩变小的八种压缩方法介绍

怎么把图片压缩变小&#xff1f;在当今这个信息高度共享的时代&#xff0c;图片不仅仅是简单的视觉元素&#xff0c;它们承载着我们的记忆、故事和创意。无论是旅行的风景、家庭的聚会&#xff0c;还是工作中的项目展示&#xff0c;图片都在我们的生活中扮演着不可或缺的角色。…

帕金森患者必看!这5种水果成“抗抖”小能手,吃出健康好生活!

在这个快节奏的时代&#xff0c;健康成为了我们最宝贵的财富之一。而对于帕金森病患者而言&#xff0c;如何在日常生活中通过合理的饮食来缓解症状、提升生活质量&#xff0c;成为了许多家庭关注的焦点。今天&#xff0c;就让我们一起探索那些被誉为“抗抖”小能手的水果&#…

Pandas -----------------------基础知识(一)

目录 Series对象 属性和方法 布尔值列表获取Series对象中部分数据 运算 DateFrame对象 常用属性 常见方法 运算 总结 Series对象 是DataFrame的列对象或者行对象 生成Series对象生成索引使用元组创建Series对象使用字典创建Series对象 通过Pandas创建对象 自定义索引 …

RealityCapture1.4设置成中文

RealityCapture 1.4 设置成中文的教程 RealityCapture 1.4 是一款强大的三维建模软件&#xff0c;它能够从图像或激光扫描中创建实景三维模型和正射影像等。以下是一个详细的教程&#xff0c;指导您如何将 RealityCapture 1.4 的界面设置为中文。 1.找到设置按钮 在WORKFLOW…

【一起学NLP】Chapter1-基本语法与神经网络的推理

备注&#xff1a;本专栏为个人的NLP学习笔记&#xff0c;欢迎大家共同讨论交流学习。代码同步&#xff1a;https://github.com/codesknight/Learning-NLP-Together 参考书籍&#xff1a;《深度学习进阶&#xff1a;自然语言处理》——斋藤康毅 目录 基础知识点复习测试环境使用…

OceanMesh2D | 基于精确距离的沿海海洋/浅水流动模型二维自动网格生成MATLAB工具箱推荐

Precise distance-based two-dimensional automated mesh generation toolbox intended for coastal ocean/shallow water flow models OceanMesh2D | 基于精确距离的沿海海洋/浅水流动模型二维自动网格生成MATLAB工具箱推荐 1. 简介2. 特点3. 代码基本要求:4. 基本流程 1. 简…

一、Numpy使用

1、numpy的简单使用 import numpy as np #利用as给numpy起一个别名np# 使用array来承接这个数组 array np.array([[1,2,3],[2,3,4]])print(array) print("number of dim:", array.ndim) # ndim 数组维度 print("shape:", array.shape) # 数组的形…

c++编程(27)——IO流(1)

欢迎来到博主的专栏&#xff1a;c编程 博主ID&#xff1a;代码小豪 文章目录 标准IO流标准输入输出cincin的四个状态标志 标准IO流 <iostream>库中的IO类在之前已经经常使用了&#xff0c;但是我们还从未仔细的了解过。在c标准库中&#xff0c;IO类分为两种&#xff0c;…

sqli-labs靶场自动化利用工具——第2关

文章目录 概要整体架构流程技术细节执行效果小结 概要 Sqli-Labs靶场对于网安专业的学生或正在学习网安的朋友来说并不陌生&#xff0c;或者说已经很熟悉。那有没有朋友想过自己开发一个测试脚本能实现自动化化测试sqli-labs呢&#xff1f;可能有些人会说不是有sqlmap&#…

摄影社团管理系统

基于springbootvue实现的摄影社团管理系统 &#xff08;源码L文ppt&#xff09;4-075 第四章 系统概要设计 4.1系统设计原理 设计原理是指系统的设计来源&#xff0c;它将需求合理地分解为功能&#xff0c;并抽象地描述系统的模块和其下的功能。在功能模块化后&#xff…

【数据结构】排序算法---桶排序

文章目录 1. 定义2. 算法步骤3. 演示3.1 动态演示13.2 动态演示23.3 图片演示13.4 图片演示2 4. 性质5. 算法分析6. 代码实现C语言PythonJavaCGo 结语 1. 定义 桶排序&#xff08;英文&#xff1a;Bucket sort&#xff09;是计数排序的升级版&#xff0c;适用于待排序数据值域…

【算法】最长公共子序列(C/C++)

最长公共子序列&#xff08;LCS&#xff0c;Longest Common Subsequence&#xff09;问题简称&#xff08;LCS&#xff09;&#xff0c;是动态规划里面里面的基础算法。它的所解决的问题是&#xff0c;在两个序列中找到一个序列&#xff0c;使得它既是第一个序列的子序列&#…