当前位置: 首页 > news >正文

c++ std::map

c++ map

C++ 中的 std::map 是一个关联容器,用于存储 键值对(Key-Value),并根据键(Key)自动排序。其底层实现基于 红黑树,因此支持高效的插入、查找和删除操作(时间复杂度为 \(O(\log n)\))。以下是 std::map 的使用简介:


1. 基本概念

  • 键值对(Key-Value):每个元素由唯一的键(Key)和对应的值(Value)组成,类似字典。
  • 有序性:键默认按升序排列(可通过自定义比较器修改)。
  • 唯一性:键必须是唯一的,不允许重。
  • 应用场景:适合需要快速查找、插入和删除键值对的场景,如数据库索引、配置映射等。

2. 定义与初始化

#include <map>
#include <string>int main() {// 定义一个 map,键为 int,值为 stringstd::map<int, std::string> myMap;// 初始化方式 1:直接赋值myMap[1] = "apple";myMap[2] = "banana";// 初始化方式 2:使用 insert 插入 pairmyMap.insert(std::pair<int, std::string>(3, "cherry"));// 初始化方式 3:使用列表初始化(C++11)std::map<int, std::string> myMap2 = {{4, "date"},{5, "elderberry"}};return 0;
}

3. 插入数据

  • 常用方法

    // 使用 operator[]
    myMap[6] = "fig";// 使用 insert 插入单个元素
    myMap.insert({7, "grape"});// 使用 emplace 直接构造元素(C++11)
    myMap.emplace(8, "honeydew");// 批量插入(通过迭代器区间)
    std::vector<std::pair<int, std::string>> vec = {{9, "kiwi"}, {10, "lemon"}};
    myMap.insert(vec.begin(), vec.end());
    

4. 查找数据

  • 查找键是否存在

    auto it = myMap.find(3);
    if (it != myMap.end()) {std::cout << "Found: " << it->second << std::endl; // 输出 "cherry"
    } else {std::cout << "Key not found!" << std::endl;
    }
    
  • 获取值(推荐使用 at() 避免意外插入)

    try {std::cout << myMap.at(3) << std::endl; // 输出 "cherry"
    } catch (const std::out_of_range& e) {std::cout << "Key not found!" << std::endl;
    }
    

5. 删除数据

  • 按键删除

    myMap.erase(3); // 删除键为 3 的元素
    
  • 按迭代器删除

    auto it = myMap.find(4);
    if (it != myMap.end()) {myMap.erase(it);
    }
    
  • 批量删除

    myMap.erase(5, 7); // 删除键在 [5,7) 范围内的元素(C++20)
    

6. 遍历 map

  • 使用迭代器

    for (auto it = myMap.begin(); it != myMap.end(); ++it) {std::cout << "Key: " << it->first << ", Value: " << it->second << std::endl;
    }
    
  • 使用范围 for 循环(C++11)

    for (const auto& [key, value] : myMap) {std::cout << "Key: " << key << ", Value: " << value << std::endl;
    }
    
  • 反向遍历

    for (auto it = myMap.rbegin(); it != myMap.rend(); ++it) {std::cout << "Key: " << it->first << ", Value: " << it->second << std::endl;
    }
    

7. 其他常用操作

  • 判断键是否存在

    bool exists = myMap.count(3) > 0; // count 返回 0 或 1
    
  • 获取大小与清空

    std::cout << "Size: " << myMap.size() << std::endl;
    myMap.clear(); // 清空所有元素
    
  • 合并两个 map(C++17)

    std::map<int, std::string> map1 = {{1, "a"}, {2, "b"}};
    std::map<int, std::string> map2 = {{3, "c"}, {2, "d"}};map1.merge(map2); // 合并 map2 到 map1,若键冲突则保留 map1 的值
    

8. 性能优化与注意事项

  1. 避免频繁插入/删除

    • 如果需要频繁修改键值对,考虑使用 std::unordered_map(哈希表实现,平均查找复杂度为 \(O(1)\))。
  2. 自定义比较器

    • 默认按键升序排序,可通过自定义比较器实现降序或其他排序规则:

      struct CompareDesc {bool operator()(int a, int b) const { return a > b; }
      };
      std::map<int, std::string, CompareDesc> myMap; // 按键降序排序
      
  3. 避免意外插入

    • 使用 at() 代替 operator[],防止访问不存在的键时自动插入默认值。
  4. 批量插入优化

    • 使用 emplace_hint 提供插入位置的提示,减少红黑树调整次数:

      auto hint = myMap.begin();
      myMap.emplace_hint(hint, 11, "orange");
      
  5. 内存管理

    • 避免频繁拷贝大 map,可使用引用或指针传递。

9. 常见陷阱

  • 键的不可变性:键的值不能直接修改(否则会破坏红黑树的排序结构)。若需修改键,需先删除旧键,再插入新键。
  • 默认值插入operator[] 在键不存在时会插入默认值,可能导致意外行为。
  • 自定义类型的键:若键为自定义类型,需重载比较运算符或提供自定义比较器。

10. 与 std::unordered_map 的对比

特性 std::map std::unordered_map
底层实现 红黑树(有序) 哈希表(无序)
查找复杂度 \(O(\log n)\) 平均 \(O(1)\),最坏 \(O(n)\)
插入/删除复杂度 \(O(\log n)\) 平均 \(O(1)\),最坏 \(O(n)\)
是否保持有序
适用场景 需要有序性或范围查询 需要高效率的随机访问

通过合理选择 std::mapstd::unordered_map,结合具体需求优化操作方式,可以高效地处理键值对数据。

http://www.hskmm.com/?act=detail&tid=576

相关文章:

  • 调一调
  • 【稳定检索|线上线下参会|马理工主办】第十一届建筑、土木与水利工程国际学术会议(ICACHE 2025)
  • 软考架构备考-面向对象
  • 在AI技术快速实现创意的时代,挖掘真实需求成为核心竞争力——某知名实时语音转录系统需求洞察
  • 多版本jdk环境下,指定jdk版本执行jar文件
  • 【Python】Word文档解析表格并导出Excel
  • 海明码破解指南:从让人头疼的软考题到内存背后的无名英雄
  • 2025.9.9 总结
  • 基于Python+Vue开发的医院门诊预约挂号系统源码+运行
  • 2025 9 6 总结
  • 2025.9.8 总结
  • OLE读取EXCEL内容
  • 第5篇、 Kafka 数据可靠性与容错机制
  • Shell符号详解
  • Dynamics 365 CRM + Power Platform 技术顾问:解锁 IT 高薪赛道,长沙爱码士 IT 助你全程通关
  • Rope pbds
  • 295、嫦娥
  • 25.9.8随笔联考总结
  • rmrs 题解
  • IT 失业人员的福音:借微软 Dynamics 365 CRM 与 Power Platform 快速重启职业生涯
  • qoj10096 Generating Random Trees
  • 测试
  • PHP 轻松处理千万行数据 内存不爆,服务器不卡
  • 2025 杭电暑期多校训练
  • 友链
  • BongoCat - 可爱的桌面互动猫咪
  • qoj6279 Honeycomb
  • Vue 将api 获取的 json 数据保存到本地
  • Claude Code新手入门指南:AI编程助手完全教程
  • 0124_观察者模式(Observer)