深入理解C++关联式容器:set、multiset、map和multimap详解

序列式容器 与 关联式容器

我们知道:

  • C++ 中,我们将 vector、list、queue 这种底层为线性序列的数据结构叫做 序列式容器,其存储的就是元素本身
  • 关联式容器键-值对的形式存储数据。每个键在容器中必须是唯一的,而值则与相应的键关联。

键-值对

<key,value> 结构,在C++标准库中,std::pair 是一个模板类,其存储的就是键值对。其定义如下:

namespace std {
    template <class T1, class T2>
    struct pair {
        typedef T1 first_type;
        typedef T2 second_type;

        T1 first;
        T2 second;

        pair();
        pair(const T1& x, const T2& y);
        template<class U, class V> pair(const pair<U, V>& p);
    };
}

一般键是唯一且不可重复的标识符,用于查找和访问对应的值。值可以是任何数据类型,包括整数、字符串、对象等


树形结构的关联式容器

C++ 中,标准库提供了 std::map 、std::set 、std::multimap、std::multimap 四个基于红黑树实现的关联式容器。下面依次介绍其性质与使用:


std::set

Cplusplus 官网set文档

性质

上面的cpp文档详细的介绍了set这个容器,下面我们对其性质进行总结概括:

  1. 有序性set 中的元素按照其key值自动被排序。默认情况下使用元素的 < 操作符(小于)比较,也可以通过自定义比较函数来指定排序规则。插入 / 删除 新元素时,set 会自动确保元素的有序性,并且不会插入重复的元素。

  2. 唯一性:对于set,元素的值(value)即是键(key),而且每个value必须是唯一的。

  3. 不可修改性:set中的元素不能在容器中修改(元素总是const),但是可以从容器中插入或删除它们。

  4. 底层实现:通常基于红黑树实现,因此插入、删除和查找操作的平均时间复杂度都是 O(log n)。

  5. 迭代器支持:set 提供了正向迭代器,可以用于遍历集合中的元素。

Attention:

  • set在创建时为std::set<value> myset;,只存放value,其底层上存放的是<value, value> 的键值对。实际实现中,只存储了键,而值部分为空。
  • 我们知道set中的元素不可直接修改,为什么?
    • 由于 std::set 是一种基于红黑树实现的关联容器,它要求元素在容器中保持有序性和唯一性。因此,直接修改元素可能会破坏红黑树的结构,导致容器不再符合要求。

使用

模板参数列表

下面是C++标准库中的 set 模板参数列表

template<
    class Key, // Key(键)类型
    class Compare = std::less<Key>, // 比较函数类型
    class Allocator = std::allocator<Key> // 分配器类型
> class set;

分别解释:

  • Keystd::set 中存储的元素的类型。set 中的元素按照键值自动被排序,并且每个元素在中都是唯一的
  • Compare可选的比较函数类型,用于指定元素的比较规则。默认情况下,使用 < 操作符进行比较。
  • 可选的分配器类型用于分配内存和管理容器中的元素

应用

数据去重std::set会自动对元素进行排序并移除重复的元素

std::vector<int> numbers = {4, 1, 2, 2, 3, 4};
std::set<int> uniqueNumbers(numbers.begin(), numbers.end());
// 现在uniqueNumbers中包含的是去重后的元素{1, 2, 3, 4}

快速查找std::set内部基于红黑树实现,因此查找操作非常高效,时间复杂度为O(log n)

std::set<std::string> names = {"Alice", "Bob", "Charlie", "David"};
if (names.find("Bob") != names.end())  // 查找Bob
{
    std::cout << "Bob 在set中" << std::endl;
}

迭代器的使用可以通过迭代器来访问集合中的元素

#include <iostream>
#include <set>

int main() {
    std::set<int> mySet = {3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5};

    // 使用迭代器遍历 set 中的所有元素
    for (auto it = mySet.begin(); it != mySet.end(); ++it) {
        std::cout << *it << " ";
    }
    std::cout << std::endl;

    // 反向迭代器
    for (auto rit = mySet.rbegin(); rit != mySet.rend(); ++rit) {
        std::cout << *rit << " ";
    }
    std::cout << std::endl;

    // 使用迭代器查找特定元素
    int target = 5;
    auto findIt = mySet.find(target);
    if (findIt != mySet.end()) {
        std::cout << target << "找到了" << std::endl;
    } else {
        std::cout << target << "没找到" << std::endl;
    }

    return 0;
}

std::map

cplusplus 官网文档

性质

上面的cpp文档详细的介绍了map这个容器,下面我们对其性质进行总结概括:

  1. 有序性map 中的元素按照key的大小顺序进行排序,默认情况下采用升序方式。这使得在 map 中进行范围遍历时,能够以键的顺序访问元素。

  2. 唯一键map 中的键是唯一的,如果插入一个已经存在的键,则会覆盖原有的键对应的值。

  3. 查找效率:由于 map 基于红黑树实现,对于元素的查找、插入和删除操作具有较高的效率,时间复杂度为 O(log n)。

  4. 键-值对映射:map 提供了键和值之间的映射关系,通过键可以快速查找对应的值,这使得 map 适合用于需要快速查找和检索值的场景。

Attention:

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

模板参数列表

template <class Key, class T, class Compare = std::less<Key>, class Allocator = std::allocator<std::pair<const Key, T>>>
class map;

这里是对每个模板参数的解释:

  • Key键的类型,用于唯一标识每个元素。

  • T值的类型,与键相关联的数据类型。

  • Compare可选参数,用于比较键的函数对象类型,默认情况下采用 std::less<Key>,表示采用 < 操作符进行键的比较。

  • Allocator可选参数,内存分配器的类型,默认情况下采用 std::allocator<std::pair<const Key, T>>,表示使用标准分配器

应用

我们下面用map进行一些应用:字典、计数器

字典

#include <iostream>
#include <map>
#include <string>

int main() {
	// 创建字典
    std::map<std::string, std::string> dictionary;

    // 添加词条到字典
    dictionary["apple"] = "苹果";
    dictionary["banana"] = "香蕉";
    dictionary["cat"] = "猫";

    // 查找词条
    std::string word = "";
    getline(word);
    if (dictionary.find(word) != dictionary.end()) {
        std::cout << word << ": " << dictionary[word] << std::endl;
    } else {
        std::cout << "字典中未查到该词" << std::endl;
    }

    return 0;
}

上面的代码中,我们使用map<string, string> 来模拟字典,key,value分别对应一个单词的英文和中文含义。

计数器

#include <iostream>
#include <map>

int main() {
    std::map<int, int> counter;

    // 统计数字出现的次数
    int numbers[] = {5, 3, 7, 5, 3, 9, 5};
    for (int num : numbers) {
        counter[num]++;
    }

    // 输出数字及其出现的次数
    for (const auto& pair : counter) {
        std::cout << pair.first << " occurs " << pair.second << " times" << std::endl;
    }

    return 0;
}

计数器通过将value值设为int类型,可以统计值为key的元素的出现个数。


std::multiset

cpluscplus官网文档

性质

multiset 的 模板参数列表 与 set 是相同的,但在性质上有所差别。

这里我们先看 多重集(Multiset)和集合(Set)之间 的区别

  1. 元素重复性multiset 允许元素重复出现,而 set 中每个元素都是唯一的。

  2. 插入操作:在 set 中,插入已经存在的元素会被忽略,因为元素不能重复;而在 multiset 中,可以插入重复的元素,每个元素将会被单独计数

  3. 查找操作:在 set 中,由于元素唯一,查找特定元素的速度较快;而在 multiset 中,由于元素可能重复,查找特定元素需要遍历多个相同元素

  4. 应用场景:set 适合用于需要保持元素唯一性的场景,比如存储不重复的关键字;而 multiset 则适用于需要统计元素重复次数的场景,比如统计数据集中各个元素的出现频率。

使用

当有以下情况时,我们会使用 multiset 而不是set:

  1. 需要存储重复元素的情况
  2. 不需要维护元素的唯一性
  3. 需要快速查找和删除重复元素

下面的代码展示了,multiset的使用:

int main() {
    std::multiset<int> numbers;

    // 插入元素
    numbers.insert(10);
    numbers.insert(20);
    numbers.insert(30);
    numbers.insert(20);  // 插入重复的元素

    // 输出元素
    std::cout << "该multiset包含: ";
    for (const auto& num : numbers) {
        std::cout << " " << num;
    }
    std::cout << std::endl;

    // 统计特定元素的重复次数
    int target = 20;
    int count = numbers.count(target);
    printf("值为%d的元素共有%d个\n", target, count);

    // 删除指定元素
    numbers.erase(target);

    // 再次输出元素
    printf("删除元素%d后,multiset共包含:", target);
    for (const auto& num : numbers) {
        std::cout << " " << num;
    }
    std::cout << std::endl;

    return 0;
}

代码输出结果:

在这里插入图片描述


std::multimap

cpluscplus官网文档

性质

这里我们主要关心 multimap 与 map 的区别(性质 及 操作):

  • 键的唯一性map中的键是唯一的,每个键只能对应一个值。而multimap允许多个键有相同的值,一个键可以对应多个值

  • 插入顺序: map按照键的大小顺序进行排序,并保持该顺序。multimap不对键进行排序,插入的顺序被保留

  • 查找操作在map中,由于键的唯一性,使用find()函数查找一个键时,如果找到了就返回该键对应的值,如果没找到则返回end()迭代器。而在multimap中,find()函数返回指向第一个匹配的键-值对的迭代器,如果没有匹配的键,则返回end()迭代器。

  • 删除操作: 在map中,使用erase()函数删除一个键时,如果该键存在,则删除该键对应的键-值对,并将其从容器中移除。而在multimap中,erase()函数会删除所有与给定键匹配的键-值对

总的来说,multimap 提供了一种允许重复键并自动排序的数据结构,适用于需要按照键进行检索且允许重复键的情况下使用。

应用

我们利用multimap 一个key可以对应多个value的特性,下面的示例使用 multimap 构建了一个学生表:

#include <iostream>
#include <map>
#include <string>

int main() {
    std::multimap<std::string, std::string> studentCourses;

    // 添加学生和他们的课程
    studentCourses.insert(std::make_pair("Alice", "Math"));
    studentCourses.insert(std::make_pair("Alice", "Physics"));
    studentCourses.insert(std::make_pair("Bob", "Chemistry"));
    studentCourses.insert(std::make_pair("Bob", "Biology"));
    studentCourses.insert(std::make_pair("Bob", "Math"));

    // 输出每个学生选修的课程
    for (const auto& pair : studentCourses) {
        std::cout << pair.first << " takes " << pair.second << std::endl;
    }

    return 0;
}

输出如下:

在这里插入图片描述

结尾

上面介绍的关联式容器,在做OJ题时也会经常用上,在理解其使用场景后做题,可以很好的运用它们。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Windows没有USB启动选项很常见,但解决方法更常见

当试图在计算机上重新安装Windows 11/10操作系统,或从安装介质启动时,一些用户看到错误–系统没有任何USB启动选项,请在启动管理器菜单中选择其他启动选项。此错误出现在不同OEM的多个设备,原因包括启用了安全引导、禁用了Legacy/CSM支持、联想服务引擎、未正确制作可引导U…

本地化小程序运营 同城小程序开发

时空的限制让本地化的线上平台成为一种追求&#xff0c;58及某团正式深挖人们城镇化、本地化的信息和商业需求而崛起的平台&#xff0c;将二者结合成本地化小程序&#xff0c;显然有着巨大的市场机会。本地化小程序运营可以结合本地化生活需求的一些信息&#xff0c;以及激发商…

linux下使用Docker Compose部署Spug实现公网远程访问

&#x1f4d1;前言 本文主要是linux下使用Docker Compose部署Spug实现公网远程访问的文章&#xff0c;如果有什么需要改进的地方还请大佬指出⛺️ &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是青衿&#x1f947; ☁️博客首页&#xff1a;CSDN主页放风讲故事 &am…

vcomp120.dll丢失怎么办?vcomp120.dll丢失的解决方法分享

vcomp120.dll丢失”。这个错误通常会导致某些应用程序无法正常运行&#xff0c;给用户带来困扰。那么&#xff0c;当我们遇到这个问题时&#xff0c;应该如何修复呢&#xff1f;下面我将为大家介绍四个修复vcomp120.dll丢失的方法。 一、使用dll修复程序修复 可以通过百度或许…

【PWN · heap | unlink | free_hook】[SUCTF 2018 招新赛]unlink

在前期学习了unlink后&#xff0c;今天翻NSSCTF找到一道名为unlink的题目&#xff0c;尝试不看wp做。过程很顺利&#xff01; 前言 题目对于知识点unlink还是非常裸的&#xff0c;很直接&#xff0c;思路很清晰。 一、题目 二、思路浅析 通过对该程序的反编译&#xff0c;我们…

前端案例-css实现ul中对li进行换行

场景描述&#xff1a; 我想要实现&#xff0c;在展示的item个数少于4个的时候&#xff0c;则排成一行&#xff0c;并且均分&#xff08;比如说有3个&#xff0c;则每个的宽度为33.3%&#xff09;&#xff0c;如果item 个数大于4&#xff0c;则进行换行。 效果如下&#xff1a…

4.0 Linux进程前导知识

个人主页&#xff1a;Lei宝啊 愿所有美好如期而遇 冯.诺依曼体系 CPU&#xff1a;运算器&#xff0c;控制器 输入设备&#xff1a;键盘&#xff0c;麦克风&#xff0c;摄像头&#xff0c;鼠标&#xff0c;网卡&#xff0c;磁盘等。 输出设备&#xff1a;显示器&#xff0…

KMP算法理论

KMP算法理论 前缀&#xff1a;包含首字母不包含尾字母的都称为前缀 例如 前缀 后缀&#xff1a;只包含尾字母不包含首字母的的称为后缀 后缀 寻找最长相等的前缀和后缀 前缀表 所谓next数组就是前缀表&#xff0c;在遇到冲突时next数组会告诉我们要回退到哪里 next数组的不同…

Java基础-基础语法

1、概述 一个 Java 程序可以认为是一系列对象的集合&#xff0c;而这些对象通过调用彼此的方法来协同工作。 对象&#xff1a;对象是类的一个实例&#xff0c;有状态和行为。例如&#xff0c;一条狗是一个对象&#xff0c;它的状态有&#xff1a;颜色、名字、品种&#xff1b;…

No189.精选前端面试题,享受每天的挑战和学习

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6 🍨 阿珊和她的猫_CSDN个人主页 🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入…

十三、W5100S/W5500+RP2040树莓派Pico<FTP Server>

文章目录 1. 前言2. 相关简介2.1 简述2.2 原理2.3 优点2.4 应用 3. WIZnet以太网芯片4. FTP Server运行测试4.1 程序流程图4.2 测试准备4.3 连接方式4.4 相关代码4.5 测试现象 5. 注意事项6. 相关链接 1. 前言 在当今的信息化时代&#xff0c;互联网已经成为人们生活、工作不可…

轻量级 SSO 方略:基于 OIDC 规范(二)

上一篇文章介绍了 SSO 相关的基础数据&#xff0c;这样有了 ClientId 和密钥后&#xff0c;我们就要准备客户端这边的代码。客户端当前指的便是一个网站&#xff08;也就是 RP&#xff09;&#xff0c;这个网站要求有会员功能&#xff0c;典型地网站导航上通常会有“注册”或“…

深度学习之基于YoloV5电梯电动车预警系统

欢迎大家点赞、收藏、关注、评论啦 &#xff0c;由于篇幅有限&#xff0c;只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 深度学习在电梯电动车预警系统中的应用是一个复杂的系统工程&#xff0c;涉及计算机视觉、机器学习、深度学习等领域…

CLIP:万物分类(视觉语言大模型)

本文来着公众号“AI大道理” ​ 论文地址&#xff1a;https://arxiv.org/abs/2103.00020 传统的分类模型需要先验的定义固定的类别&#xff0c;然后经过CNN提取特征&#xff0c;经过softmax进行分类。然而这种模式有个致命的缺点&#xff0c;那就是想加入新的一类就得重新定义…

14:00面试,14:08就出来了,问的问题有点变态

从小厂出来&#xff0c;没想到在另一家公司又寄了。 到这家公司开始上班&#xff0c;加班是每天必不可少的&#xff0c;看在钱给的比较多的份上&#xff0c;就不太计较了。没想到8月一纸通知&#xff0c;所有人不准加班&#xff0c;加班费不仅没有了&#xff0c;薪资还要降40%,…

【C++】函数指针 ④ ( 函数指针做函数参数 | 使用函数指针间接调用函数 | 函数指针做参数 | 函数指针类型的本质 | 函数指针做参数意义 )

文章目录 一、函数指针做函数参数1、使用函数指针间接调用函数2、函数指针做参数3、函数指针类型的本质4、函数指针做参数意义 二、代码示例 - 函数指针做函数参数 一、函数指针做函数参数 1、使用函数指针间接调用函数 在上一篇博客 【C】函数指针 ③ ( 函数指针语法 | 函数名…

说说React Jsx转换成真实DOM过程?

一、是什么 react通过将组件编写的JSX映射到屏幕,以及组件中的状态发生了变化之后 React会将这些「变化」更新到屏幕上 在前面文章了解中,JSX通过babel最终转化成React.createElement这种形式,例如: <div> < img src="avatar.png" className="…

VS Code设置技巧

基础设置 中文界面 安装扩展&#xff1a;Chinese(Simplified) Language Pack 自动换行 文件 - 首选项 - 设置&#xff0c;搜索wrap&#xff0c;找到Editor: Word Wrap&#xff0c;将其更改为on。

【数据结构】非递归实现二叉树的前 + 中 + 后 + 层序遍历(听说面试会考?)

&#x1f466;个人主页&#xff1a;Weraphael ✍&#x1f3fb;作者简介&#xff1a;目前学习C和算法 ✈️专栏&#xff1a;数据结构 &#x1f40b; 希望大家多多支持&#xff0c;咱一起进步&#xff01;&#x1f601; 如果文章对你有帮助的话 欢迎 评论&#x1f4ac; 点赞&…

如何在OpenWrt上部署uhttpd搭建web服务器,并实现公网远程访问

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…