- 5
- 0
- 约1.78千字
- 约 3页
- 2016-11-28 发布于北京
- 举报
与在底层的区别分析
++i与i++在底层的区别分析 计算机中应用程序(Application)的内存布局 在我们学习编程语言的过程中熟练掌握自己编写的程序的内存布局是十分重要的,“掌握了内存就掌握了一切”。我们知道我们编写的程序(源代码)要经过编译器(compiler)的编译才可以变成能够在计算机上执行的程序(C++是.exe可执行文件,Java中是编译成字节码(.class)文件才能在虚拟计算机(JWM)上运行)。鉴于此我们就知道编译器为我们的程序编制做了大量的工作,但是却隐藏了从源程序到目标程序转换的过程,很多时候就会出现许多的困惑,所以我建议大家不要死抠编程语言的语法规则,而是要试着从内存的角度分析程序的运行。在这里主要使用VC++的VS2010强大的调试功能分析一下基本原理。 首先,我们需要知道程序运行要使用的4大内存区域:代码段(code segment)、数据段(data segment)、堆栈(stack)、堆(heap)。他们的作用分别如下: 代码段(code segment):存储程序运行的代码 数据段(data segment):Java中的字符串常量、静态变量(使用static关键字修饰的) 堆栈(stack):局部变量 堆(heap):Java中使用new关键字申请的 注意:在Java中除了基础数据类型中的4类8种数据类型,其他的数据类型就是引用类型(类、接口、数组),需要引起注意的是如下的Java语句: int [] score = {89, 79, 76}; //静态初始化数组 有过C++编程经验的程序员知道它在线程堆栈中分配,然而在Java中数组作为引用数据类型它会占用两块内存,如下所示: 图1 一维基础数据类型数组内存结构 之所以我们在Java的语法书上看到上面的那一句int型静态初始化数组等价于: int [] score = new int[]{89, 79, 76}; 是因为我们写的“int [] score = {89, 79, 76};”Java编译器会做类似上面的替换,而实际中的内存布局依然是上面那样的。 如果想打好Java的基础建议观看马士兵主讲的Java基础,尤其是里面的内存分析是最为精辟的。 ++i与i++在底层的区别分析 问题的引出(C++代码截图): 图2 C++中的代码实现 程序运行情况: 图3 C++中的代码实现运行结果 在图2的断点处,和同样都是对i进行赋值但是i的结果却不一样。为什么了,如果我们仅仅从语法上讲大家不是很明白但是我们从汇编的角度来考虑就豁然开朗了。 图4 i = (i2++)对应的汇编 i = (i2++)的反汇编代码分析: 004141D3:将dword ptr [i2](即i2中的内存单元)中的数据拷贝到eax寄存器中 004141D6:将eax寄存器中的数据(即i2)拷贝到dword ptr [i](即i的内存单元)中 004141D9:将dword ptr [i2](即i2中的内存单元)中的数据拷贝到ecx寄存器中 004141DC:将ecx寄存器中的内容自增1 004141DF:将ecx寄存器中的内容(ecx加1后的数据)拷贝到dword ptr [i2] 图5 i =(++i2)对应的汇编 i =(++i2)的反汇编代码分析:将dword ptr [i2](即i2中的内存单元)中的数据拷贝到eax寄存器中 0041425C:将eax寄存器中的内容自增1 0041425F:将eax寄存器中的内容(eax加1后的数据)拷贝到dword ptr [i2](即i2中的内存单元)将dword ptr [i2](即i2中的内存单元)中的数据拷贝到ecx寄存器中将ecx寄存器中的内容(ecx加1后的数据)拷贝到dword ptr [i](即i所在的内存单元) 通过以上的分析我们清晰的了解了i =(++i)和i = (i++)在底层实现的区别。
有哪些信誉好的足球投注网站
文档评论(0)