一篇搞懂C++ STL 存储重复键值对容器std::multimap

2024-08-30 21:28

本文主要是介绍一篇搞懂C++ STL 存储重复键值对容器std::multimap,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
    • 为什么使用 `std::multimap`
    • `std::multimap` 与 `std::map` 的区别
    • 字符串图表示区别
    • `std::multimap` 的构造函数和操作函数
      • 构造函数
      • 成员函数
    • 示例代码
  • 总结


前言

std::multimap 是 C++ STL 中的一个关联容器,用于存储键值对。与 std::map 不同,std::multimap 允许一个键关联多个值。这种特性使得 std::multimap 在需要存储重复键的场景中非常有用,例如实现字典或索引时需要存储多个条目。


为什么使用 std::multimap

  • 允许重复键std::multimap 允许多个元素具有相同的键,这使得它在需要存储多个具有相同属性的条目时非常有用。
  • 有序存储:与 std::map 一样,std::multimap 保持元素的键有序,支持高效的键查找。
  • 高效插入和查找std::multimap 提供对键的高效插入和查找操作,适合用作需要频繁查找的场景。

std::multimapstd::map 的区别

std::multimapstd::map 都是关联容器,但它们在存储键值对时有以下不同点:

  • 允许重复键

    • std::multimap:允许多个具有相同键的元素。
    • std::map:不允许有重复的键,每个键对应唯一的值。
  • 存储结构

    • std::multimap:键值对是有序的,但多个相同键的元素会按照插入顺序排列。
    • std::map:键值对是有序的,每个键只能出现一次。

字符串图表示区别

std::multimap                        std::mapKey: 1        1        2             Key: 1        2
Value: A   Value: B   Value: C    Value: A   Value: B// No duplicate keys allowed

std::multimap 的构造函数和操作函数

构造函数

  • 默认构造函数

    std::multimap();
    
    std::multimap<int, std::string> m;  // 创建一个空的 std::multimap
    
  • 范围构造函数

    template<class InputIterator>
    std::multimap(InputIterator first, InputIterator last);
    
    std::pair<int, std::string> arr[] = { {1, "one"}, {2, "two"}, {1, "uno"} };
    std::multimap<int, std::string> m(std::begin(arr), std::end(arr));
    
  • 拷贝构造函数

    std::multimap(const std::multimap& other);
    
    std::multimap<int, std::string> m1 = {{1, "one"}, {2, "two"}};
    std::multimap<int, std::string> m2(m1);  // 拷贝构造
    
  • 移动构造函数

    std::multimap(std::multimap&& other) noexcept;
    
    std::multimap<int, std::string> m1 = {{1, "one"}, {2, "two"}};
    std::multimap<int, std::string> m2(std::move(m1));  // 移动构造
    

成员函数

  • insert

    std::pair<iterator, bool> insert(const value_type& value);
    iterator insert(iterator hint, const value_type& value);
    template <class InputIterator>
    void insert(InputIterator first, InputIterator last);
    
    std::multimap<int, std::string> m;
    m.insert(std::make_pair(1, "one"));    // 插入单个元素
    m.insert(m.end(), std::make_pair(2, "two")); // 插入指定位置
    std::pair<int, std::string> arr[] = { {1, "uno"}, {2, "dos"} };
    m.insert(std::begin(arr), std::end(arr));  // 插入范围
    
  • erase

    size_type erase(const key_type& key);
    iterator erase(iterator position);
    iterator erase(iterator first, iterator last);
    
    std::multimap<int, std::string> m = {{1, "one"}, {1, "uno"}, {2, "two"}};
    m.erase(1);  // 删除所有键为1的元素
    m.erase(m.find(2));  // 删除键为2的元素
    
  • find

    iterator find(const key_type& key);
    
    std::multimap<int, std::string> m = {{1, "one"}, {2, "two"}};
    auto it = m.find(1);
    if (it != m.end()) {std::cout << "Found: " << it->second << std::endl;
    }
    
  • count

    size_type count(const key_type& key) const;
    
    std::multimap<int, std::string> m = {{1, "one"}, {1, "uno"}, {2, "two"}};
    std::cout << "Count of key 1: " << m.count(1) << std::endl;
    
  • equal_range

    std::pair<iterator, iterator> equal_range(const key_type& key);
    std::pair<const_iterator, const_iterator> equal_range(const key_type& key) const;
    
    std::multimap<int, std::string> m = {{1, "one"}, {1, "uno"}, {2, "two"}};
    auto range = m.equal_range(1);
    for (auto it = range.first; it != range.second; ++it) {std::cout << "Value: " << it->second << std::endl;
    }
    
  • clear

    void clear() noexcept;
    
    std::multimap<int, std::string> m = {{1, "one"}, {2, "two"}};
    m.clear();  // 清空所有元素
    

示例代码

下面的示例展示了 std::multimap 的各种构造函数和操作函数的用法:

#include <iostream>
#include <map>
#include <string>int main() {// 使用默认构造函数std::multimap<int, std::string> m1;// 使用值构造函数m1.insert(std::make_pair(1, "one"));m1.insert(std::make_pair(1, "uno"));m1.insert(std::make_pair(2, "two"));// 使用范围构造函数std::pair<int, std::string> arr[] = { {1, "one"}, {2, "two"}, {1, "uno"} };std::multimap<int, std::string> m2(std::begin(arr), std::end(arr));// 使用拷贝构造函数std::multimap<int, std::string> m3(m2);// 使用移动构造函数std::multimap<int, std::string> m4(std::move(m3));// 插入元素m4.insert(std::make_pair(3, "three"));// 查找元素auto it = m4.find(1);if (it != m4.end()) {std::cout << "Found key 1: " << it->second << std::endl;}// 计数元素std::cout << "Count of key 1: " << m4.count(1) << std::endl;// 遍历所有键值对std::cout << "Contents of multimap:" << std::endl;for (const auto& pair : m4) {std::cout << pair.first << ": " << pair.second << std::endl;}// 删除元素m4.erase(1);// 清空元素m4.clear();return 0;
}

总结

std::multimap 是一个非常有用的 STL 容器,特别适合需要存储重复键值对的场景。它与 std::map 的主要区别在于允许存储多个相同键的元素,并且保持元素的有序性。通过理解 std::multimap 的构造函数和操作函数的用法,可以更有效地处理需要重复键的场景,同时享受 STL 容器提供的高效插入和查找操作的优势。

这篇关于一篇搞懂C++ STL 存储重复键值对容器std::multimap的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1121974

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java JUC并发集合详解之线程安全容器完全攻略

《JavaJUC并发集合详解之线程安全容器完全攻略》Java通过java.util.concurrent(JUC)包提供了一整套线程安全的并发容器,它们不仅是简单的同步包装,更是基于精妙并发算法构建... 目录一、为什么需要JUC并发集合?二、核心并发集合分类与详解三、选型指南:如何选择合适的并发容器?在多

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

k8s搭建nfs共享存储实践

《k8s搭建nfs共享存储实践》本文介绍NFS服务端搭建与客户端配置,涵盖安装工具、目录设置及服务启动,随后讲解K8S中NFS动态存储部署,包括创建命名空间、ServiceAccount、RBAC权限... 目录1. NFS搭建1.1 部署NFS服务端1.1.1 下载nfs-utils和rpcbind1.1

python语言中的常用容器(集合)示例详解

《python语言中的常用容器(集合)示例详解》Python集合是一种无序且不重复的数据容器,它可以存储任意类型的对象,包括数字、字符串、元组等,下面:本文主要介绍python语言中常用容器(集合... 目录1.核心内置容器1. 列表2. 元组3. 集合4. 冻结集合5. 字典2.collections模块

一文详解MySQL索引(六张图彻底搞懂)

《一文详解MySQL索引(六张图彻底搞懂)》MySQL索引的建立对于MySQL的高效运行是很重要的,索引可以大大提高MySQL的检索速度,:本文主要介绍MySQL索引的相关资料,文中通过代码介绍的... 目录一、什么是索引?为什么需要索引?二、索引该用哪种数据结构?1. 哈希表2. 跳表3. 二叉排序树4.