从运算符重载深入理解自增自减运算符

在学习C++的过程中经常会遇到一些自增运算符的问题,比如

1.后缀自增运算不能赋值

int tmp = 0;
tmp++ = 10;//error 不能做左值

以过往经验来看该语句意为对自增后的 tmp进行赋值,但为什么 tmp++后会变成右值呢?本文将通过自定义实现 int来理解使用自增运算符过程中出现的问题 。

class CInt
{
public:
	CInt(int val = 0)
	{
		data = val;
	}
	CInt& operator ++()//++CInt
	{
		++data;
		return *this;
	}
	const CInt operator ++(int t)//CInt++
	{
		const CInt val = *this;
		++*this;
		return val;
	}
	/*
	operator int()//实现类型转换的语句
	{
		return data;
	}
	*/
private:
	int data;
};
前置++与后置++的内部实现不同,前置++实现自增后,返回引用后置++先保存当前对象的副本,实现自增操作后,返回副本。

前置++

CInt& operator ++()//前置++
{
	++data;
	return *this;
}
参数:      void

返回值:  引用(CInt&),返回当前对象要使用引用,如果返回类型为CInt,返回的对象为临时量。

CInt a ;//a.data = 0
++(++a);//a.data = 1, ++a 返回临时量

后置++

const CInt operator ++(int t)//后置++
{
	const CInt val = *this;
	++*this;
	return val;
}

参数 :  int,为与区分前置++,后置++接受一个整形的形参,一般编译器将会把0作为实参; 


返回值:与内置类型保持一致,内置类型临时变量不能做左值,返回类型const CInt。

在计算效果相同的情况下,前置++效率更高,后置++会产生临时量,调用了一次构造和析构函数。


2.在vs中进行四则运算时,前缀自增优先级最高,后缀自增运算最低。

int b = 3;
int c = (b++)+(b++)+(++b);//??c = 12

分析VS产生的汇编代码:


分析一下重点代码可以发现,先进行前缀自增运算,再进行加法运算结果写入c中,即c=4+4+4 ,然后进行后缀自增运算。


3.printf 函数中,前缀自增运算用一个寄存器存放 i的值,最终返回为 i的值,后缀自增运算返回当前 i的值。

int i = 0;
printf("%d,%d,%d",i++,++i,i++);//?? 2 3 0
 

printf函数从右向左一次运算,最右边的i++返回当前i(0)的值,在进行++运算,再中间的++i运算后,然后最左边的 i++返回当前i(2)的值,在进行++运算,最后把 i(3)的值返回给中间的++i。



c++ 前置++与后置++的区别 - 语行 - 博客园

关于C语言前置++和后置++的二三事_luv8818_4BB_新浪博客

C++_运算符重载 总结 - ArrowToMe - 博客园

(转)前置++和后置++的区别 - balingybj - 博客园

前置(后置)++ /--与+/-在不同编译器下运算方式不同 - CSDN博客

发布了26 篇原创文章 · 获赞 16 · 访问量 2万+

猜你喜欢

转载自blog.csdn.net/Teemo_king/article/details/79253178