C++容器对比:map与unordered_map全解析

C++关联式容器详解:mapsetunordered_map

一、核心原理
  1. mapset(有序容器)

    • 数据结构 :基于红黑树(自平衡二叉搜索树)
    • 特性
      • 元素按严格弱排序 (默认<)自动排序
      • 插入/删除/查找时间复杂度:O(\\log n)
    • map :存储键值对 (key, value)
    • set :仅存储唯一键 key
  2. unordered_map(无序容器)

    • 数据结构 :基于哈希表(散列表)
    • 特性
      • 元素顺序不可控(依赖哈希函数)
      • 平均插入/删除/查找时间复杂度:O(1)
      • 最坏情况:O(n)(哈希冲突严重时)
二、关键操作对比
操作 map/set unordered_map
插入 insert() insert()
查找 find() find()
删除 erase() erase()
遍历 有序迭代 无序迭代
自定义排序 支持比较器 不支持
自定义哈希 不支持 需定义哈希函数
三、典型应用场景
  1. map示例:词频统计
cpp 复制代码
#include <map>
#include <string>

std::map<std::string, int> wordCount;
for (const auto& word : words) {
    wordCount[word]++; // 自动初始化值为0
}
  1. set示例:去重排序
cpp 复制代码
#include <set>
#include <vector>

std::vector<int> nums = {3, 1, 4, 1, 5};
std::set<int> uniqueSet(nums.begin(), nums.end()); // 输出:{1, 3, 4, 5}
  1. unordered_map示例:缓存系统
cpp 复制代码
#include <unordered_map>

std::unordered_map<std::string, Data> cache;
if (cache.find(key) == cache.end()) {
    cache[key] = fetchData(); // 快速查找
}
四、性能优化要点
  1. map/set

    • 自定义比较器提升效率
    cpp 复制代码
    struct CaseInsensitiveCompare {
        bool operator()(const std::string& a, const std::string& b) const {
            return std::tolower(a[0]) < std::tolower(b[0]);
        }
    };
    std::map<std::string, int, CaseInsensitiveCompare> customMap;
  2. unordered_map

    • 预分配桶数量减少冲突
    cpp 复制代码
    std::unordered_map<int, std::string> preallocMap;
    preallocMap.reserve(1000); // 预分配空间
    • 自定义哈希函数
    cpp 复制代码
    struct CustomHash {
        size_t operator()(const MyClass& obj) const {
            return std::hash<int>()(obj.id) ^ std::hash<std::string>()(obj.name);
        }
    };
五、选择策略
  1. 有序遍历范围查询 :选map/set
  2. 追求极致查找速度不要求顺序 :选unordered_map
  3. 内存敏感场景:红黑树(map/set)通常比哈希表更省内存

注意unordered_map的迭代器失效规则与map不同,插入操作可能导致全部迭代器失效(桶重建时)。

相关推荐
郝学胜_神的一滴13 小时前
CMake 034:生成器表达式:解耦构建时序、精简分支逻辑的终极利器
c++·cmake
见过夏天1 天前
C++ 基础入门完全指南
c++
用户805533698033 天前
不止三件套:QObject 属性系统全关键字与运行时反射!
c++·qt
BadBadBad__AK3 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
卷无止境4 天前
Eigen 库如何借助 OpenMP 加速计算
c++·后端
卷无止境4 天前
OpenMPI、MPICH 与 OpenMP:关系、核心概念与架构全解
c++·后端
郝学胜_神的一滴5 天前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake
卷无止境7 天前
C++ 的Eigen 库全解析
c++
卷无止境7 天前
现代 C++特性大盘点:一门脱胎换骨的老语言
c++·后端
郝学胜_神的一滴7 天前
CMake 27:缓存变量的特性、语法、类型与实操全解
c++·cmake