今天在阅读《google c++ 编程风格》的文档的时候,5.10. 前置自增和自减:有一句话引起了我的注意:

对于迭代器和其他模板对象使用前缀形式 (++i) 的自增, 自减运算符.,理由是 前置自增 (++i) 通常要比后置自增 (i++) 效率更高。于是我查了查前置++和后置++的区别。

注意:《more effective c++》条款8也专门叙述了问题。后来我发现,下面的文章基本就是它的翻版,哈哈

前置++和后置++的区别

  1. int a = 0;
  2. ++ a;   //前置++
  3. a++;    //后置++

《C专家编程》中有如下描述(P276,人民邮电出版社):

++a表示取a的地址,增加它的内容,然后把值放在寄存器中;

a++表示取a的地址,把它的值装入寄存器,然后增加内存中的a的值;

另外,网上找了篇文章,通过从运算符重载的角度来探讨他们的不同,如下:

假设有一个类Age,描述年龄。该类重载了前置++和后置++两个操作符,以实现对年龄的自增。

  1. class Age
  2. {
  3. public:
  4. Age& operator++() //前置++
  5. {
  6. ++i;
  7. return *this;
  8. }
  9. const Age operator++(int) //后置++
  10. {
  11. Age tmp = *this;
  12. ++(*this);  //利用前置++
  13. return tmp;
  14. }
  15. Age& operator=(int i) //赋值操作
  16. {
  17. this->i = i;
  18. return *this;
  19. }
  20. private:
  21. int i;
  22. };

从上述代码,我们可以看出前置++和后置++,有3点不同:

  1. 返回类型不同
  2. 形参不同
  3. 代码不同
  4. 效率不同

返回值类型的区别

前置++的返回类型是Age&,后置++的返回类型const Age。这意味着,前置++返回的是左值,后置++返回的是右值。(关于左值和右值的讨论很多,见本文下面)

左值和右值,决定了前置++和后置++的用法。

  1. int main()
  2. {
  3. Age a;
  4. (a++)++;  //编译错误
  5. ++(a++);  //编译错误
  6. a++ = 1;   //编译错误
  7. (++a)++;  //OK
  8. ++(++a);  //OK
  9. ++a = 1;   //OK
  10. }

++的类型是const Age,自然不能对它进行前置++、后置++、赋值等操作。

++a的类型是Age&,当然可以对它进行前置++、后置++、赋值等操作

a++的返回类型为什么要是const对象呢?

有两个原因:

  1. 如果不是const对象,a(++)++这样的表达式就可以通过编译。但是,其效果却违反了我们的直觉 。a其实只增加了1,因为第二次自增作用在一个临时对象上。
  2. 另外,对于内置类型,(i++)++这样的表达式是不能通过编译的。自定义类型的操作符重载,应该与内置类型保持行为一致 。

a++的返回类型如果改成非const对象,肯定能通过编译,但是我们最好不要这样做。

++a的返回类型为什么是引用呢?

这样做的原因应该就是:与内置类型的行为保持一致。前置++返回的总是被自增的对象本身。因此,++(++a)的效果就是a被自增两次。

形参的区别

前置++没有形参,而后置++有一个int形参,但是该形参也没有被用到。很奇怪,难道有什么特殊的用意?

其实也没有特殊的用意,只是为了绕过语法的限制。

前置++与后置++的操作符重载函数,函数原型必须不同。否则就违反了“重载函数必须拥有不同的函数原型”的语法规定。

虽然前置++与后置++的返回类型不同,但是返回类型不属于函数原型。为了绕过语法限制,只好给后置++增加了一个int形参。

原因就是这么简单,真的没其他特殊用意。其实,给前置++增加形参也可以;增加一个double形参而不是int形参,也可以。只是,当时就这么决定了。

代码实现的区别

前置++的实现比较简单,自增之后,将*this返回即可。需要注意的是,一定要返回*this。

后置++的实现稍微麻烦一些。因为要返回自增之前的对象,所以先将对象拷贝一份,再进行自增,最后返回那个拷贝。

在Age的代码中,后置++利用了前置++来实现自增。这样做是为了避免“自增的代码”重复。

在本例中,自增的代码很简单,就是一行++i,没有必要这样做。但是在其它自增逻辑复杂的例子中,这么做还是很有必要的。

效率的区别

如果不需要返回自增之前的值,那么前置++和后置++的计算效果都一样。但是,我们仍然应该优先使用前置++,尤其是对于用户自定义类型的自增操作。

前置++的效率更高,理由是:后置++会生成临时对象。

从Age的后置++的代码实现也可以看出这一点。

  1. const Age operator++(int) //后置++
  2. {
  3. Age tmp = *this;
  4. ++(*this);  //利用前置++
  5. return tmp;
  6. }

很明显,tmp是一个临时对象,会造成一次构造函数和一次析构函数的额外开销。虽然,编译器在某些情况下可以优化掉这些开销。但是,我们最好不要依赖编译器的行为。

所以,在非内置类型的时候,尽量使用前置++,因为效率高(后置自增,效率低)

转载于:https://www.cnblogs.com/cthon/p/9185263.html

C++之运算符重载(前置++和后置++)相关推荐

  1. C++_类和对象_C++运算符重载_递增运算符重载_重载++运算符_前置++_后置++重载---C++语言工作笔记057

    然后我们再来看这个递增运算符的重载. 我们知道,递增运算符,有前置++,和后置++,可以看到,上面写了 然后我们这次就自己定义一个int的类型,MyInteger,然后实现,递增,递减的操作. 首先我 ...

  2. c++重载自增与自减运算符(前置与后置)

    运算符重载 要点: 1. 后置的运算符要带一个整型参数(用来与前置运算符区分开). 2.后置的返回值不要用引用(否则会因为是局部变量导致返回为一个不存在的值). 3.再在置中要定义一个临时变量来返回之 ...

  3. 一元运算符重载 前置和后置++ --(这种一般用成员函数来实现重载)

    #include <iostream>using namespace std; //实现一元运算符的前置重载 class A { public:friend A operator++(A ...

  4. # c++运算符重载之 前置++, 后置++, 负号运算符, 类型转换函数, 以及输入输出运算符...

    c++运算符重载之 前置++, 后置++, 负号运算符, 类型转换函数, 以及输入输出运算符 标签(空格分隔): c++ 前言 我在c++学习的过程中, 对这几个不太常见的运算符重载不太会写.出现了很 ...

  5. C++回顾之前置++、后置++、不等号!及赋值运算符重载

    运算符重载的主要目的是为了让类对象能像普通数据类型一样能够进行加减乘除,自加自减等操作,非常直观方便.现在来回顾C++的自加减(分前置与后置)以及不等号非运算符,赋值运算符的重载. 1 ++重载 (1 ...

  6. JavaScript运算符:递增递减运算符前置和后置的区别

    从两段代码说起 var num1 = 2; var num2 = 20; var num3 = --num1 + num2; var num4 = num1 + num2; console.log(n ...

  7. (转)前置++和后置++的区别

    今天在阅读<google c++ 编程风格>的文档的时候,5.10. 前置自增和自减:有一句话引起了我的注意: 对于迭代器和其他模板对象使用前缀形式 (++i) 的自增, 自减运算符.,理 ...

  8. 前置++和后置++的区别

    今天在阅读<google c++ 编程风格>的文档的时候,5.10. 前置自增和自减:有一句话引起了我的注意: 对于迭代器和其他模板对象使用前缀形式 (++i) 的自增, 自减运算符.,理 ...

  9. 前置++与后置++之一道简单的题目引发的思考

    引言 昨晚一时兴起,我脑子就问自己下面的代码会输出什么,也不知道我脑子为什么有这个代码模型,只是模糊的有些印象: #include <stdio.h> #include <stdli ...

最新文章

  1. python+requests+unittest+excel_接口自动化测试 unittest+request+excel(踩‘坑’)
  2. 关于Two pointers的个人理解
  3. openwrt下如何只编译linux内核
  4. 一、为了OFFER系列 | 阿里云天池赛在线编程:移动的圆
  5. JSONPATH使用方法
  6. POI导出人事报表:代码实现
  7. 分数优先遵循志愿php源码_分数优先 遵循志愿
  8. centos删除gnome_自定义你的 GNOME 桌面主题 | Linux 中国
  9. git 更换本地目录_git 本地库的使用
  10. leetcode-reverse words in a string
  11. 何时适合进行自动化测试?(上)
  12. sqllite开发安卓项目_安卓学习笔记(一)
  13. Ubuntu16.04源码安装postgresql-9.6.6数据库
  14. java 单例模式(饿汉模式和懒汉模式)
  15. matlab和r语言做热图,R语言画图与MATLAB画图PK
  16. python爬取起点中文网小说_爬虫实战——起点中文网小说的爬取
  17. 联想服务器改win7系统教程,手把手教你联想win11改win7系统教程
  18. 静态页面练习(pc和移动端两套)
  19. 抖音从流量上游杀入本地生活市场,美团如何“以牙还牙”?
  20. Pytorch教程[03]transforms

热门文章

  1. USACO Section1.2 Your Ride Is Here(水题)
  2. Python3 写JSON数据到文件(多行格式化版)
  3. order by + 数字的用意
  4. Windows编程—获取操作系统默认浏览器路径
  5. artTemplate
  6. Java 异常练习
  7. 一种用javascript实现的比较兼容的回到顶部demo + 阻止事件冒泡
  8. CentOS 6.4 命令行 安装 VMware Tools
  9. 三星 平板手机电脑 Galaxytab2忘记开机密码解决方法
  10. ASP.NET中实现MSN通知消息功能