(++a)+=(a++)到底发生了什么

我们可以利用VC编译器的一个选项/FAs来生成对应的汇编代码。/FAs还会在汇编代码中加入注释注明和C++代码的对应关系,十分有助于分析。在VS2008中可以这样打开/FAs:

Build代码,可以在输出目录下发现对应的.ASM文件。

通过一段代码及其汇编代码来看看计算机到底如何处理了(++a)+=(a++),c++源代码为:

#include <iostream>
using namespace std;
int main()
{
	int a=4;
	a+=(a++);
	cout<<a<<endl;
	return 0;
}
对应的.asm文件中为:

; 6    : 	(++a)+=(a++);

	mov	eax, DWORD PTR _a$[ebp]
	add	eax, 1
	mov	DWORD PTR _a$[ebp], eax
	mov	ecx, DWORD PTR _a$[ebp]
	add	ecx, DWORD PTR _a$[ebp]
	mov	DWORD PTR _a$[ebp], ecx
	mov	edx, DWORD PTR _a$[ebp]
	add	edx, 1
	mov	DWORD PTR _a$[ebp], edx
可以发现,计算机首先将a的值加1,然后计算a+=a(此时结果为10),最后在进行的计算++(此时为11);

也即是说,++a这样的语句是在整句话执行之前完成的,而a++这样的语句是在这句话完成之后才进行的;不妨对代码进行修改,为:

#include <iostream>
using namespace std;
int main()
{
	int a=4;
	(++a)+=(++a);
	cout<<a<<endl;
	return 0;
}
对应的汇编代码为:

; 6    : 	(++a)+=(++a);

	mov	eax, DWORD PTR _a$[ebp]
	add	eax, 1
	mov	DWORD PTR _a$[ebp], eax
	mov	ecx, DWORD PTR _a$[ebp]
	add	ecx, 1
	mov	DWORD PTR _a$[ebp], ecx
	mov	edx, DWORD PTR _a$[ebp]
	add	edx, DWORD PTR _a$[ebp]
	mov	DWORD PTR _a$[ebp], edx
这段代码可以证实我们的猜测。

一个更复杂一点的程序:

#include <iostream>
using namespace std;
int main()
{
	int a=4;
	int b=3;
	int c=2;
	(++a)+=(++a)+(c++)+(++b);
	cout<<a<<endl;
	cout<<b<<endl;
	cout<<c<<endl;
	return 0;
}

对应的汇编代码为:

; 8    : 	(++a)+=(++a)+(c++)+(++b);

	mov	eax, DWORD PTR _a$[ebp]
	add	eax, 1
	mov	DWORD PTR _a$[ebp], eax
	mov	ecx, DWORD PTR _a$[ebp]
	add	ecx, 1
	mov	DWORD PTR _a$[ebp], ecx
	mov	edx, DWORD PTR _b$[ebp]
	add	edx, 1
	mov	DWORD PTR _b$[ebp], edx
	mov	eax, DWORD PTR _a$[ebp]
	add	eax, DWORD PTR _c$[ebp]
	add	eax, DWORD PTR _b$[ebp]
	add	eax, DWORD PTR _a$[ebp]
	mov	DWORD PTR _a$[ebp], eax
	mov	ecx, DWORD PTR _c$[ebp]
	add	ecx, 1
	mov	DWORD PTR _c$[ebp], ecx
同样如此~

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值