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

c++ map作为返回值_详解 C++ STL 中 map::erase 的正确姿势

(给CPP开发者加星标,提升C/C++技能)

来源:liuzhi67 https://blog.csdn.net/liuzhi67/article/details/50950843

【导读】:本文主要讲解map erase迭代器的正确操作。

之前在代码中使用map::erase函数时,误搬了vector::erase的用法,导致Server down掉了,好在在测试环境就及时发现了问题,在上线前进行了补救==。以下总结一下map::erase的正确用法。首先看一下在循环中使用vector::erase时我习惯的用法:

for(vector<int>::iterator it = vecInt.begin(); it != vecInt.end();)
{
    if(*it == 0)
    {
        it = vecInt.erase(it);
    }
    else
    {
        it++;
    }
}

程序从一个vector中删除值为0的元素,利用了vector::erase函数根据iterator删除某个元素时会返回下一个元素的iterator的性质:http://www.cplusplus.com/reference/vector/vector/erase/

C++98

iterator erase (iterator position);

这一种用法是没有问题的。

然而当想当然的在map::erase上照搬上面erase的用法时,就有问题了,查看http://www.cplusplus.com/reference/map/map/erase/ 上的说明:

C++98

(1) 
     void erase (iterator position);
(2) 
size_type erase (const key_type& k);
(3) 
     void erase (iterator first, iterator last);

如上所示,C++98中map::erase并没有返回值为iterator的原型函数。

那么问题来了it=map.erase(it),然后对it进行操作会发生什么呢?会发生传说中的“未定义的行为”!包括但不限于程序挂掉、机器死机、地球地震、宇宙毁灭等–原因是什么呢?在执行map.erase(it)之后,it这个iterator已经失效了,考虑C语言中一个失效释放了的指针,再次引用它会导致什么问题呢?

在循环中正确使用map::erase的方法是什么呢?如下:

for(map<int,int>::iterator it = mapInt.begin(); it != mapInt.end();)
{
    if(it->second == 0)
    {
        mapInt.erase(it++);
    }
    else
    {
        it++;
    }
}

在网上找mapInt.erase(it++)的说明,比较详细的一种解释为:

http://blog.csdn.net/lmh12506/article/details/9167653

该方法中利用了后缀++的特点,这个时候执行mapInt.erase(it++);这条语句分为三个过程

  • 先把it的值赋值给一个临时变量做为传递给erase的参数变量

  • 因为参数处理优先于函数调用,所以接下来执行了it++操作,也就是it现在已经指向了下一个地址。

  • 再调用erase函数,释放掉第一步中保存的要删除的it的值的临时变量所指的位置。

然而个人感觉比较费解,意思是第一步先把it的值传给了函数调用的形参,然后又回去执行i+1的操作吗?这样总感觉it++的执行被硬生生的切成了两部分,只能硬记住这一结论。

直到后来看了《STL源码剖析》中的++i和i++实现方式的区别,然后某一天,再看到《More Effective C++》里的说明,突然开窍了,mapInt.erase(it++)的机理终于不再神秘。

其实在mapInt.erase(it++)中,it++确实是作为一个完整的执行过程,it++的具体实现代码其实类似以下:

// postfix form: fetch and increment
map<int, int>::iterator operator++(int)//通过一个多余的int参数与prefix++区分
{
    map<int, int>::iterator tmp = *this; // fetch
    increment(); // increment,map内部由红黑树实现,此函数负责指向下一个有序元素的iterator
    return tmp; // return what was
}

上面代码的最终返回的值其实是tmp,tmp存储的是*this的旧值,this后来通过increment函数自增了,但是tmp的依然保持原值,最后将tmp返回赋值作为erase的参数,所以在mapInt.erase(it++)中,其实it++是作为一个整体执行完成了的,在传值给erase函数之前,it其自身其实已经+1了,不过后缀++返回的却是一个未执行+1操作的旧值,所以后面erase函数依然删除的是原it位置的值,同时该迭代器失效,然而之前it已经+1自增过了,所以不受其影响噢。

关于上面代码中调用的前缀++代码类似如下:

// prefix form: increment and fetch
map<int, int>::iterator& operator++()
{
    increment(); // increment
    return *this; // fetch
}

也正因为后缀++会比前缀++的操作多一个临时变量,并且其是以传值复制的方式返回给调用方,所以一般而言后缀++的效率会比前缀++效率低一些。

值得一提的是,在最新的C++11标准中,已经新增了一个map::erase函数执行后会返回下一个元素的iterator,然而不知道啥时候C++11才能达到现在C++98的覆盖程度,谨慎一点还是使用map.erase(it++)比较保险。

http://www.cplusplus.com/reference/map/map/erase/

C++11

(1) 
iterator  erase (const_iterator position);
(2) 
size_type erase (const key_type& k);
(3) 
iterator  erase (const_iterator first, const_iterator last);

最后,有的小伙伴可能会问为啥前缀++和后缀++的返回值一个是迭代器引用,一个却是迭代器传值?简单来说,前缀++返回的便是传参进来的迭代器,自然可以返回迭代器本身的引用,然而后缀++返回的是一个函数内部的临时变量,在函数执行完后便析构了,必然不能传引用。注意既然是通过传值的方式返回,对其返回值的修改对于原it是没有影响的,举例来说(it++)++的结果其实it只自增了一次,第二次++只是对其(it++)的返回值执行了++,对原it没有任何效果。

- EOF -

推荐阅读   点击标题可跳转

1、无锁队列的实现

2、C++ std::function技术浅谈

3、做引擎开发,更需要深入 C++ 内存管理

关于 map erase,欢迎在评论中和我探讨。觉得文章不错,请点赞和在看支持我继续分享好文。谢谢!

关注『CPP开发者』

看精选C++技术文章 . 加C++开发者专属圈子

↓↓↓

94abaa7fb26383978219fcc638bff9e7.png

点赞和在看就是最大的支持❤️

相关文章:

  • python试卷(有答案版本、个人答案不是官方答案)_python试卷(有答案版本,个人答案不是官方答案)(精品文档)_共7页...
  • echarts 饼图进度条_Echarts实现环状半圆形饼图
  • 多开脚本_现阶段魔兽世界怀旧服晚上脚本成灾?G币会暴跌吗?
  • c#split方法拆分为数据_C#:使用String.Split方法在每个单独的行中拆分字符串列表?...
  • 压缩图片_图片压缩
  • python随机数应用_Python中随机数的使用于详细讲解
  • 生成图片_ThinkPHP5 动态生成图片缩略图
  • c++max函数怎么用_比函数还强大的Excel分列技巧,你知道怎么用吗
  • mysql2005错误_SQL Server 2005 还原数据库错误解决方法
  • mysql中取字符串函数是_MySQL 字符串函数:字符串截取
  • mysql查询时间提前五天_MySQL查询不含周末的五天前的日期
  • elasticsearch 嵌入式_3.JanusGraph+HBase+ElasticSearch配置
  • mysql数据库怎末导出函数_MySQL数据库之mysql 导入导出数据库以及函数、存储过程的介绍...
  • Mysql 数组放进表里_新人求大神教教:如何把一个一维数组存入mysql 表格中
  • rabbitmq支持两个消费者同时提取数据吗_认识RabbitMQ从这篇文章开始
  • 3.7、@ResponseBody 和 @RestController
  • AzureCon上微软宣布了哪些容器相关的重磅消息
  • create-react-app做的留言板
  • java8-模拟hadoop
  • mysql 5.6 原生Online DDL解析
  • Redis提升并发能力 | 从0开始构建SpringCloud微服务(2)
  • 从零搭建Koa2 Server
  • 看图轻松理解数据结构与算法系列(基于数组的栈)
  • 如何优雅的使用vue+Dcloud(Hbuild)开发混合app
  • 深度解析利用ES6进行Promise封装总结
  • 项目实战-Api的解决方案
  • - 语言经验 - 《c++的高性能内存管理库tcmalloc和jemalloc》
  • # 20155222 2016-2017-2 《Java程序设计》第5周学习总结
  • # include “ “ 和 # include < >两者的区别
  • (C语言版)链表(三)——实现双向链表创建、删除、插入、释放内存等简单操作...
  • (done) 两个矩阵 “相似” 是什么意思?
  • (function(){})()的分步解析
  • (NSDate) 时间 (time )比较
  • (附源码)apringboot计算机专业大学生就业指南 毕业设计061355
  • (附源码)ssm捐赠救助系统 毕业设计 060945
  • (附源码)计算机毕业设计高校学生选课系统
  • (七)c52学习之旅-中断
  • (已解决)什么是vue导航守卫
  • (转) 深度模型优化性能 调参
  • (转)shell中括号的特殊用法 linux if多条件判断
  • ****** 二十三 ******、软设笔记【数据库】-数据操作-常用关系操作、关系运算
  • ***汇编语言 实验16 编写包含多个功能子程序的中断例程
  • .gitignore文件_Git:.gitignore
  • .net core 客户端缓存、服务器端响应缓存、服务器内存缓存
  • .net mvc actionresult 返回字符串_.NET架构师知识普及
  • .NET 应用启用与禁用自动生成绑定重定向 (bindingRedirect),解决不同版本 dll 的依赖问题
  • .NET8.0 AOT 经验分享 FreeSql/FreeRedis/FreeScheduler 均已通过测试
  • .NET中的十进制浮点类型,徐汇区网站设计
  • :O)修改linux硬件时间
  • @PreAuthorize注解
  • [] 与 [[]], -gt 与 > 的比较
  • []T 还是 []*T, 这是一个问题
  • [ABC294Ex] K-Coloring
  • [AutoSar]BSW_Memory_Stack_003 NVM与APP的显式和隐式同步
  • [C#]使用DlibDotNet人脸检测人脸68特征点识别人脸5特征点识别人脸对齐人脸比对FaceMesh