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

C++中std::map容器中元素删除方法汇总 - 详解

C++中std::map容器中元素删除方法汇总 - 详解

1. std::map 的基本特点

  • 有序关联容器,按 key 排序存储。
  • 内部通常为 红黑树(平衡二叉搜索树)。
  • key 唯一,value 可修改。
  • 查找、插入、删除复杂度均为 O(log n)

2. 删除元素的基本方法

2.1 使用 key 删除

std::map<
int, std::string> m = {
{
1,"a"
}, {
2,"b"
}, {
3,"c"
}
};
// 删除 key=2 的元素
size_t removed = m.erase(2);
// 返回删除元素数量(0 或 1)
  • 复杂度:O(log n)
  • 注意:如果 key 不存在,返回 0。
  • 适用场景:知道要删除的元素 key。

2.2 使用迭代器删除单个元素

auto it = m.find(3);
if(it != m.end()) {
m.erase(it);
}
  • 复杂度:O(log n)(查找)+ O(1)(删除节点)
  • 优点:可以在迭代过程中安全删除元素(需注意 ++it 避免失效)
  • 注意erase(it) 会使该迭代器失效,但其他迭代器不受影响。

2.3 使用迭代器范围删除

// 删除 key 从 1 到 2(不包含 2)
m.erase(m.begin(), m.find(2));
  • 复杂度:O(log n + k),k 为删除元素个数

  • 注意

    • 左闭右开区间 [first, last)
    • 删除范围过大时,仍是高效操作,因为底层红黑树会做局部重平衡。

2.4 清空整个 map

m.clear();
// 删除所有元素
  • 复杂度:O(n)
  • 效果:调用每个元素的析构函数,释放内存。

3. 删除元素的迭代安全写法

在遍历 map 时删除元素,需要小心迭代器失效问题:

3.1 正确写法

for(auto it = m.begin(); it != m.end();
) {
if(it->second == "b") {
it = m.erase(it);
// erase 返回下一个有效迭代器
} else {
++it;
}
}
  • 核心erase 返回指向下一个元素的迭代器,避免使用已失效的迭代器。
  • 错误示例
for(auto it = m.begin(); it != m.end();
++it) {
if(it->second == "b") {
m.erase(it);
// 错误!it 失效,下一步 ++it 未定义行为
}
}

4. 删除元素时的复杂度分析

方法时间复杂度备注
erase(key)O(log n)单元素删除
erase(iterator)O(1) 删除节点 + O(log n) 重平衡已知迭代器,删除效率高
erase(iterator_first, iterator_last)O(k + log n)k 为删除元素个数
clear()O(n)遍历析构每个元素

小结:若要删除大量元素,使用范围删除比单个 key 删除更高效。


5. 高级技巧

5.1 条件删除(lambda + erase_if)

C++20 提供 std::erase_if

#include <map>#include <algorithm>std::map<int,std::string> m = {{1,"a"}, {2,"b"}, {3,"c"}};// 删除 value == "b" 的元素std::erase_if(m, [](auto &pair){return pair.second == "b";});
  • 简化遍历删除操作
  • 底层安全迭代器处理
  • 兼容 STL 容器,包含 vector/map/set

5.2 使用 find + 条件删除

auto it = m.find(2);
if(it != m.end() && it->second == "b") {
m.erase(it);
}
  • 适用于复杂条件判断

6. 内存与性能注意点

  1. erase 会调用元素的析构函数,复杂对象可能开销大。

  2. 如果大量删除后还要插入新元素,map 内部可能会频繁重平衡,考虑:

    • 使用 unordered_map(哈希表)删除复杂度均摊 O(1)
    • 批量删除而不是循环单个删除
  3. 对于大数据 SLAM 系统,频繁删除 map 中点云或特征索引时,可以:

    • 使用 内存池 存储 value,避免频繁分配/释放
    • 或使用 std::vector + 索引表 做批量逻辑删除,提高缓存局部性

7. std::map 的各种删除元素综合示例

示例1
下面展示 std::map 的各种删除元素方法,包括:按 key 删除、按 iterator 删除、按条件删除(比如满足某个条件的元素),并配合一些遍历操作来看效果。

#include <iostream>#include <map>#include <string>int main() {// 创建一个 std::mapstd::map<int, std::string> myMap = {{1, "apple"},{2, "banana"},{3, "cherry"},{4, "date"},{5, "elderberry"}};std::cout <<"初始 map:" << std::endl;for (const auto&[key, value] : myMap) {std::cout << key <<" -> " << value << std::endl;}// 方法1:按 key 删除std::cout <<"\n删除 key=2 的元素" << std::endl;myMap.erase(2);// 返回值是删除的元素数量for (const auto&[key, value] : myMap) {std::cout << key <<" -> " << value << std::endl;}// 方法2:按 iterator 删除std::cout <<"\n删除第一个元素" << std::endl;auto it = myMap.begin();myMap.erase(it);for (const auto&[key, value] : myMap) {std::cout << key <<" -> " << value << std::endl;}// 方法3:按条件删除(删除 key > 3 的元素)std::cout <<"\n删除 key > 3 的元素" << std::endl;for (auto iter = myMap.begin(); iter != myMap.end();) {if (iter->first >3) {iter = myMap.erase(iter);// erase 返回下一个有效 iterator} else {++iter;}}for (const auto&[key, value] : myMap) {std::cout << key <<" -> " << value << std::endl;}// 方法4:清空整个 mapstd::cout <<"\n清空整个 map" << std::endl;myMap.clear();std::cout <<"map 大小: " << myMap.size() << std::endl;return 0;}

输出示例:

初始 map:
1 -> apple
2 -> banana
3 -> cherry
4 -> date
5 -> elderberry
删除 key=2 的元素
1 -> apple
3 -> cherry
4 -> date
5 -> elderberry
删除第一个元素
3 -> cherry
4 -> date
5 -> elderberry
删除 key > 3 的元素
3 -> cherry
4 -> date
清空整个 map
map 大小: 0

示例2
下面一个更高级的示例,演示如何在循环中 安全删除满足复杂条件的多个元素,同时避免 iterator 失效问题。这里用 std::map::erase 返回的新 iterator 来保证安全。

#include <iostream>#include <map>#include <string>int main() {std::map<int, std::string> myMap = {{1, "apple"},{2, "banana"},{3, "cherry"},{4, "date"},{5, "elderberry"},{6, "fig"},{7, "grape"}};std::cout <<"初始 map:" << std::endl;for (const auto&[key, value] : myMap) {std::cout << key <<" -> " << value << std::endl;}// 高级删除:删除所有 key 为奇数或者 value 长度 > 5 的元素for (auto it = myMap.begin(); it != myMap.end();) {if (it->first % 2 != 0 || it->second.length() >5) {// erase 返回删除元素后的下一个有效 iteratorit = myMap.erase(it);} else {++it;}}std::cout <<"\n删除奇数 key 或 value 长度 > 5 的元素后:" << std::endl;for (const auto&[key, value] : myMap) {std::cout << key <<" -> " << value << std::endl;}return 0;}

输出示例:

初始 map:
1 -> apple
2 -> banana
3 -> cherry
4 -> date
5 -> elderberry
6 -> fig
7 -> grape
删除奇数 key 或 value 长度 > 5 的元素后:
4 -> date
6 -> fig

重点说明:

  1. 不要在循环中直接用 ++iterase(it) 分开,因为 erase(it) 会使当前 iterator 失效。
  2. 正确方式:用 it = myMap.erase(it)erase 会返回下一个有效 iterator。
  3. 这种方法适用于复杂条件删除,非常安全。

http://www.sczhlp.com/news/126793/

相关文章:

  • 物理半程与半时问题
  • 把自己的网站卖给别人后对方做违法关于网站建设案例
  • 深圳坪山高铁站更新备案 网站 打开
  • 景区网站建设案例拼多多运营
  • 9.22 科研小结:不要总是预设成功,失败才是常态
  • STM32光强传感器实验详解 - 实践
  • 在CodeBolcks下wxSmith的C++编程教程——从Hello world开始讲述wxSmith使用基础
  • 淮北官方网站虹桥做网站
  • 行业网站策划方案橙光游戏制作器手机版
  • 网站规范化建设米课wordpress建站
  • 网站 改域名导航滑动整屏网站
  • 如何网站切换商丘广告推广
  • 网站权重如何速度增加wordpress多站点无法访问
  • 多模态算法QwenVL、KimiVL等算法原理 - Big-Yellow
  • 多模态模型——QwenVL2.5的微调以及强化学习代码操作 - Big-Yellow
  • 从用户态到内核态:Windows CC 技术深度解析(第一篇:DNS隧道)
  • 知行网站建设兑换网站建设
  • 公司注册咨询电话优化网站定制
  • wordpress建立商业网站网站开发工具有asp
  • icp对网站内容西安网页设计师
  • 网站建设的实际价值南京软件定制
  • 好的提升设计师网站社交网站 备案
  • 字体设计教程网站做网站注意的问题
  • ppt模板下载免费版网站怎样才能被百度秒收录
  • 台州企业建站程序wordpress 主页显示
  • 如果让你建设网站之前你会想什么安徽省城乡建设网站
  • 做电影网站 需要进那些群做医疗网站颜色选择
  • 青岛哪家做网站好wordpress inc目录
  • 【Azure Batch】使用Start Task来挂载Storage Blob
  • HP notebook set your key to action key /multimedia key