从汇编来看i++与++i
故事背景,一个正在c语言的家伙,问我++i 和 i++的问题,我当时由于要去上课没给他说,正好今晚有空就測试了一下例如以下代码:
编译环境:VS2010 语言:C++

1 #include <iostream>
2 using namespace std;
3
4 int main(void)
5 {
6 int a = 1;
7 int b = 1;
8 int c;
9
10 c = a++;
11 c = ++b;
12
13 return 0;
14 }

一、我们站在汇编的角度来说明一下问题:
可能你没学过汇编,只是没关系,我们先来科普一下汇编基本知识。(我自己也不会汇编,仅仅是能看懂一些简单汇编代码)
-----------------------------------------------------------------
1)dword ptr : dword -> double word 双字节 ptr -> pointer 指针
2)mov a b : 表示将b的值赋值给a
3)add x y : 表示取x的值和y的值相加,结果再放入x中
4)另外就是cpu的8个通用寄存器 :eax, ebx, ecx, edx, esi, edi, ebp, esp
eax :是"累加器", 它是非常多加法乘法指令的缺省寄存器
ecx :是"计数器", 是反复(REP)前缀指令和LOOP指令的内定计数器
-----------------------------------------------------------------
好了,以下的汇编代码我再简单解释一下,就基本差点儿相同了。
说明:下面汇编代码解释过程中,比方:eax=1,是表示眼下eax中的值为1.

1 int a = 1;
2 00EC136E mov dword ptr [a],1 //给a赋值1
3 int b = 1;
4 00EC1375 mov dword ptr [b],1 //给b赋值1
5 int c;
6
7 c = a++;
8 00EC137C mov eax,dword ptr [a] //将a=1放入eax=1寄存器中
9 00EC137F mov dword ptr [c],eax //将eax=1放入c=1的地址中
10 00EC1382 mov ecx,dword ptr [a] //将a=1放入ecx=1寄存器中
11 00EC1385 add ecx,1 //将ecx=1和1相加,并放入ecx=2寄存器中
12 00EC1388 mov dword ptr [a],ecx //将ecx=2寄存器里的值放入a=2中
13 c = ++b;
14 00EC138B mov eax,dword ptr [b] //将b=1放入eax=1寄存器中
15 00EC138E add eax,1 //将eax=1与1相加,并放入eax=2寄存器中
16 00EC1391 mov dword ptr [b],eax //将eax=2寄存器里的值放入b=2中
17 00EC1394 mov ecx,dword ptr [b] //将b=2放入ecx=2寄存器中
18 00EC1397 mov dword ptr [c],ecx //将ecx=2寄存器里的值放入c=2中
19
20 return 0;
21 00EC139A xor eax,eax
22 }

从上面的一段汇编代码中我们能够非常清晰的看到,汇编后:
1)c = a++; 当中c的值是1,可是a中的值却已经变化为2了。
2)c = ++b; 当中c的值是2,b的值也是2。
二、以下用C++中的 ++i 与 i++ 的重载演示样例来说明一下问题:

1 /*win7_32bit,VS2010,2014年8月19日08:16:11*/
2 #include <iostream>
3 using namespace std;
4
5 class Test
6 {
7 public:
8 Test(int var) : m_var(var)
9 {}
10 //重载i++
11 const Test operator++(int)//返回const的目的在于,使"i++ = 12"这样的写法非法(注意,这里不能返回栈上的引用)
12 {
13 Test t = *this; //保存原来的数据
14 ++m_var;
15 return t; //返回原来的数据
16 }
17 //重载++i
18 Test& operator++() //为了支持"++i = 10"这样的写法,我们返回一个对象的引用
19 {
20 ++m_var;
21 return *this;
22 }
23 //重载输出流
24 friend ostream& operator<<(ostream& os, const Test& t);
25 private:
26 int m_var;
27 };
28 ostream& operator<<(ostream& os,const Test& t)
29 {
30 os<<t.m_var;
31 return os;
32 }
33
34 int main(void)
35 {
36 Test a(2);
37 Test b(3);
38 cout<<a++<<endl;//result:2
39 cout<<++b<<endl;//result:4
40 ++a = 10; //ok
41 cout<<a<<endl; //result:10
42 //b++ = 12; const 不能赋值,error
43
44 return 0;
45 }

--------------------------------------------------------------
从汇编来看i++与++i的更多相关文章
- 从汇编来看c语言之指针
一.基础研究 将下面的程序编译连接,用debug加载: 首先执行第一条语句: 发现p=(unsigned char *)0x1000;在这里是把1000赋给一个偏移地址为01af.大小为两字节的内存空 ...
- 从汇编来看c语言之变量
1.基础研究 对如图程序进行编译连接,再用debug加载. 我们在偏移地址1fa处查看main函数的内容: 执行到1fd处,发现n的偏移地址为01a6,段地址存储在ds寄存器里,为07c4. 再查看函 ...
- 从汇编来看c语言
一. 学习过程 从C语言的角度提出一些问题,这些问题再从汇编的角度考虑,还真的很有意思. (1) 我们用高级语言编程时,一般不可能不用到变量,但是一定要用到变量吗?还有这些变量从汇编的角度是怎么实现的 ...
- MSIL Hello World
最近由于需要,开始阅读 MSIL 方面的东西.我读的是<.NET 探秘——MSIL 权威指南>(<Expert .NET 2.0 IL Assembler>中译版).感觉没什么 ...
- ATPCS规则
title: ATPCS规则 tags: ARM date: 2018-10-14 17:03:23 --- ATPCS规则 ARM指令集E004armproc.chm ATPCS介绍与使用.pdf ...
- (C++)i++和++i,哪个效率高一些
在看<程序员面试笔试宝典>时,发现了这样一个问题,书中只给出了++i的效率高一些,但并没有给出具体的解释和说明. 在网上找到下面的答案: 1.从高级层面上解释 ++i 是i=i+1,表达式 ...
- arm下dlsym返回的符号地址居然不是偶对齐的。
我们都知道在写汇编函数过程都会偶对齐,而gcc编译器都会将函数编译为cpu字长对齐的地址.arm指令集是固定32位指令长度,thumb指令集是固定16位指令长度, 但是运行在arm下的程序,dlsym ...
- Go interface 原理剖析--类型转换
hi, 大家好,我是 haohognfan. 可能你看过的 interface 剖析的文章比较多了,这些文章基本都是从汇编角度分析类型转换或者动态转发.不过随着 Go 版本升级,对应的 Go 汇编也发 ...
- (七)羽夏看C语言——模板(C++)
写在前面 由于此系列是本人一个字一个字码出来的,包括示例和实验截图.本人非计算机专业,可能对本教程涉及的事物没有了解的足够深入,如有错误,欢迎批评指正. 如有好的建议,欢迎反馈.码字不易,如果本篇 ...
随机推荐
- Dynamics CRM 2013 初体验(3):新增加的功能
新系统除了修补系统历史漏洞外当然还会添加些比较有意思的新功能,至于这些新功能是否好用那就得看它是否能经过咱们这些使用者的考验了.Dynamics CRM 2013系统将不再支持Dynamics CRM ...
- cf455A Boredom
A. Boredom time limit per test 1 second memory limit per test 256 megabytes input standard input out ...
- bzoj2015 [Usaco2010 Feb]Chocolate Giving
Description Farmer John有B头奶牛(1<=B<=25000),有N(2*B<=N<=50000)个农场,编号1-N,有M(N-1<=M<=10 ...
- 【HDU 2013 猴子吃桃子】 尾递归与迭代
大一时的一道C语言练习题,可作为递归和尾递归转迭代的范例.HDU 2013 http://acm.hdu.edu.cn/showproblem.php?pid=2013 题意:猴子摘了sum个桃子,从 ...
- cURL.1 手册页
摘自http://blog.csdn.net/huangxy10/article/details/45717793 cURL.1 手册页 名称 cURL - transfer a URL 摘要 cUR ...
- linux动态库加载的秘密
摘自http://gotowqj.iteye.com/blog/1926734 摘自http://www.360doc.com/content/14/0313/13/12747488_36024641 ...
- mysql--存储过程(入门篇)
h2 { color: #fff; background-color: #7CCD7C; padding: 3px; margin: 10px 0px } h3 { color: #fff; back ...
- android 自定义圆形进度条
一.通过动画实现 定义res/anim/loading.xml如下: [html] view plain copy print ? <?xml version="1.0" ...
- Hadoop基础
Hadoop组成 包括两个核心组成:HDFS:分布式文件系统,存储海量的数据MapReduce:并行处理框架,实现任务分解和调度 搭建大型数据仓库,PB级数据的存储.处理.分析.统计等业务(搜索引擎. ...
- plaidctf2015 uncorrupt png
代码的执行时间挺长的,好囧! 参考了https://13c5.wordpress.com/2015/04/20/plaidctf-2015-png-uncorrupt/的代码 通过这个题目,也对Png ...