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不同,插入操作可能导致全部迭代器失效(桶重建时)。

相关推荐
田野追逐星光2 小时前
STL中容器list -- 讲解超详细
开发语言·c++·list
diediedei2 小时前
高性能计算通信库
开发语言·c++·算法
雾岛听蓝2 小时前
C++11新特性(可变参数模板、新的类功能、STL中的一些新变化)
开发语言·c++·经验分享·笔记
lxl13072 小时前
学习C++(6)日期类的实现+取地址运算符重载
开发语言·c++·学习
fqbqrr2 小时前
2602c++,剪切板格式
c++
Howrun7772 小时前
可调用对象
开发语言·c++
晚风吹长发2 小时前
初步了解Linux中的线程概率及线程控制
linux·运维·服务器·开发语言·c++·centos·线程
爱学习的阿磊2 小时前
模板代码跨编译器兼容
开发语言·c++·算法
带鱼吃猫3 小时前
C++STL:从 0 到 1 手写 C++ string以及高频易错点复盘
开发语言·c++