1. #include<iostream>
  2. #include<stdlib.h>
  3. using namespace std;
  4. const int &add( const int &a,const int &b)
  5. {
  6. return a+b;
  7. }
  8. int main()
  9. {
  10. const int &result = add(5,8);
  11. add(11, 8);//重复使用栈.排除虽然栈回收.但值没变的可能性
  12. add(5, 8);
  13. add(5,8);
  14. add(5,8);
  15. add(5,8);
  16. cout << result << endl;
  17. system("pause");
  18. return 0;
  19. }
   

以上结果输出13.原因在于当add的时候参数5,8.将生成2个临时对象(但这个临时对象的生存周期可伴随main).用来保存5,8
main.c 部分反汇编代码
  1. 00FC18CE mov dword ptr [ebp-0D4h],8 ;创建一个地址用来保存8
  2. 00FC18D8 mov dword ptr [ebp-0E0h],5 ;创建一个地址用来保存5
  3. 00FC18E2 lea eax,[ebp-0D4h] ;将保存8的地址赋给eax.
  4. 00FC18E8 push eax ;地址压栈
  5. 00FC18E9 lea ecx,[ebp-0E0h] ;将保存5的地址赋给ecx
  6. 00FC18EF push ecx ;将地址压入栈中
  7. 00FC18F0 call add (0FC138Eh)
  8. 00FC18F5 add esp,8 ;回收栈
  9. 00FC18F8 mov dword ptr [result],eax 

  1. const int &add(const int &a, const int &b)
  2. {
  3. 002217C0 push ebp
  4. 002217C1 mov ebp,esp
  5. 002217C3 sub esp,0CCh ;开辟局部栈空间,大小为0cch
  6. 002217C9 push ebx
  7. 002217CA push esi
  8. 002217CB push edi
  9. 002217CC lea edi,[ebp-0CCh] ;
  10. 002217D2 mov ecx,33h
  11. 002217D7 mov eax,0CCCCCCCCh
  12. 002217DC rep stos dword ptr es:[edi]
注意.以上局部栈空间大小在 esp到esp-0cch之间.当函数调用完此空间将回收(并不是真正意义的回收.只是改变栈指针罢了.原来的数值还在.不过如果此栈被反复调用(这就是为何我重复调用的原因).栈的数据当然改变)
  1. 00FC17DE mov eax,dword ptr [a] ;将临时对象5的地址赋给eax (const int &a) [a]保存的是临时对象5的
  2. 00FC17E1 mov ecx,dword ptr [eax] ;将eax保存的值给ecx
  3. 00FC17E3 mov edx,dword ptr [b] ;将临时对象b的地址赋给edx
  4. 00FC17E6 add ecx,dword ptr [edx] ;将edx保存的值加到ecx中
  5. 00FC17E8 mov dword ptr [ebp-0C8h],ecx ;将结果保存到ebp-0c8h中(临时对象)
  6. 00FC17EE lea eax,[ebp-0C8h] ;将结果地址赋给eax
  7. }
  8. 执行完Add函数后将临时对象的地址保存到result引用中
  9. 00FC18F8 mov dword ptr [result],eax 
注意了 return a+b;开辟的临时对象在ebp-0c8h处.处于局部栈之内!!!(ebp~ebp-0xcch).当add调用完后..将此回收地址赋给了result引用.于是出现了结果乱了的情况了

2.初始化一个常量引用所引发的临时对象
  当初始化一个常量引用(const reference)时,如果给定的初始化对象类型与目标引用类型不同(但是两者  能够相互转换),需要产生临时对象;
对于局部const int  temp=10这类变量.即使取地址使得其改变也不会有任何变化.
double db=10.000;
int &q=db;
以上会报错.
因为实际他会转化为 int temp=(int) db; int &a=temp;.本意是想通过a可以改变db.但实现却不可以改变(改变的是Temp).因此.编译器将会报错

当然如果是 const int &a=db;这样编译器将不会报错.因为const修饰这个temp(db)不会改变的
  1. double db = 10;
  2. 00EB1868 movsd xmm0,mmword ptr ds:[0EB6B38h]
  3. 00EB1870 movsd mmword ptr [db],xmm0
  4. const int &q = db;
  5. 00EB1875 cvttsd2si eax,mmword ptr [db] ;db的值10赋给eax
  6. 00EB187A mov dword ptr [ebp-28h],eax ;将eax赋给ebp-28h(就是临时对象)
  7. 00EB187D lea ecx,[ebp-28h]
  8. 00EB1880 mov dword ptr [q],ecx


3.修改const 常量的值
  1. #include<iostream>
  2. #include<stdlib.h>
  3. using namespace std;
  4. const int &add(const int &a, const int &b)
  5. {
  6. return a + b;
  7. }
  8. int main()
  9. {
  10. const int temp = 10;
  11. int *p = (int *)&temp;
  12. *p = 10000;
  13. cout << *p << endl;
  14. cout << temp << endl;
  15. system("pause");
  16. }

以上*p输出10000,但是temp输出10;因为const int temp=10在编译阶段就已经知道其值了
  1. cout << temp << endl;
  2. 002C4F91 push 2C1069h
  3. 002C4F96 mov edi,esp
  4. 002C4F98 push 0Ah ;将10压栈进行输出.所以..反正汇编语言是这样解释的..那就是编译器的规则了
  5. 002C4F9A mov ecx,dword ptr ds:[2CA0A8h]
  6. 002C4FA0 call dword ptr ds:[2CA0A0h]




最简单程序
  1. #include<iostream>
  2. #include<stdlib.h>
  3. using namespace std;
  4. int main()
  5. {
  6. const int &p = 1;
  7. system("pause");
  8. }
常量引用也会分配变量内存的.引用一个常量将会产生一个临时对象.临时对象赋值为1.然后将临时对象地址赋给引用.引用的底层实现其实也是一个指针常量罢了
  1. const int &p = 1;
  2. 002618E8 mov dword ptr [ebp-18h],1 ;ebp-18h是临时对象.将1赋给临时对象
  3. 002618EF lea eax,[ebp-18h] ;取出临时对象的地址
  4. 002618F2 mov dword ptr [p],eax ;将地址赋给p










反汇编角度->C++ const的更多相关文章

  1. 从汇编的角度看待const与#define

    先观察一下的代码: #include<stdio.h> int main(){ ; int y; int *pi=(int*)&i; *pi=; y=*pi; int tempi; ...

  2. 反汇编看c++引用

    继续反汇编系列,本次使用vc2008在x86体系下分析c++中的引用. 定义一个引用类型和将一个变量转换成引用类型一样吗? 引用比指针安全,真的是这样吗,对引用不理解的话比指针还危险. 为什么要用常量 ...

  3. typedef,static,const用法

    一.typedef主要功能是定义一个已存在类型的别名,但是和宏并存 宏与typedef区别 1.宏定义只是简单的字符串替换 2.typedef定义的类型是类型的别名,typedef后面是一个整体声明, ...

  4. 零基础逆向工程13_C语言07_指针01_反汇编

    1."带*类型"的特征探测 宽度 在同一个平台下,任何指针变量的尺寸都是一样的(都等于系统字长),如在32位平台中任何类型指针宽度都是32位. 声明 1.带有* 的变量类型的标准写 ...

  5. CAD图在线Web测量工具代码实现(测量距离、面积、角度等)

    CAD如今在各个领域均得到了普遍的应用并大大提高了工程技术人员的工作效率.在桌面端,AutoCAD测量工具已经非常强大:然后在Web端,如何准确.快速的对CAD图在Web进行测量呢? 功能 能Web在 ...

  6. (转)Windows驱动编程基础教程

    版权声明     本书是免费电子书. 作者保留一切权利.但在保证本书完整性(包括版权声明.前言.正文内容.后记.以及作者的信息),并不增删.改变其中任何文字内容的前提下,欢迎任何读者 以任何形式(包括 ...

  7. 【HEVC】4、HM-16.7编码一个CU(帧内部分) 3.帧内预测各种模式实现

    HEVC中一共定义了35中帧内编码预测模式,编号分别以0-34定义.其中模式0定义为平面模式(INTRA_PLANAR),模式1定义为均值模式(INTRA_DC),模式2~34定义为角度预测模式(IN ...

  8. 仿造slither.io第一步:先画条蛇

    前言 最近 slither.io 貌似特别火,中午的时候,同事们都在玩,包括我自己也是玩的不亦乐乎. 好久好久没折腾过canvas相关的我也是觉得是时候再折腾一番啦,所以就试着仿造一下吧.楼主也没写过 ...

  9. 网站启动SSL, http变为https后,session验证码错误解决方法

    网站启动SSL, http变为https后,session验证码错误解决方法   最近公司需要后台启动安全证书,证书安装完毕后,后台老提示 验证码错误,经过几天的研究,此问题已经得到有效解决,现把方法 ...

随机推荐

  1. UNIX 是啥?!和Linux什么关系?

    操作系统有两大阵营,一边是基于微软 Windows NT 的操作系统,一边是由UNIX衍生下来的操作系统. Linux, Mac OS X, Android, iOS, Chrome OS甚至路由器上 ...

  2. mysql 5.7安装密码校验插件validate_password

    在使用服务器插件之前,必须将它们加载到服务器中.MySQL支持在服务器启动和运行时加载插件.还可以在启动时控制加载插件的激活状态,并在运行时卸载它们.在加载插件时,可以从INFORMATION_SCH ...

  3. 双击内容变input框可编辑,失去焦点后修改的数据异步提交

    <html> <head> <meta charset="utf8"> <script src="https://cdn.boo ...

  4. 2018 Python开发者大调查:Python和JavaScript最配?

    在2018年秋季,Python软件基金会与JetBrains发起了年度Python开发者调查. 报告的目的是寻找Python领域的新趋势,帮助开发者深入了解2018年Python开发者的现状. 该报告 ...

  5. python GIL锁、进程池与线程池、同步异步

    一.GIL全局解释器锁 全局解释器锁 在CPython中,全局解释器锁(GIL)是一个互斥锁,它可以防止多个本机线程同时执行Python代码.之所以需要这个锁,主要是因为CPython的内存管理不是线 ...

  6. hihocoder1175 拓扑排序2

    #1175 : 拓扑排序·二 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho所在学校的校园网被黑客入侵并投放了病毒.这事在校内BBS上立刻引起了大家的讨论 ...

  7. CDH4 journalnode方式手工安装手册之二

    一.                                Hadoop配置修改 修改core-site.xml文件 <configuration> <property> ...

  8. clr(Windows 运行时和公共语言运行时)

    Windows 运行时   编译器使用 COM 引用计数机制来确定对象是否不再使用并可以删除. 因为从 Windows 运行时接口派生的对象实际上是 COM 对象,所以这是可行的. 在创建或复制对象时 ...

  9. python - 读取配置文件

    # -*- coding:utf-8 -*- ''' @project: jiaxy @author: Jimmy @file: read_config.py @ide: PyCharm Commun ...

  10. re——正则表达式

    常用的表达式规则 '.' 默认匹配除\n之外的任意一个字符,若指定flag DOTALL,则匹配任意字符,包括换行 '^' 匹配字符开头,若指定flags MULTILINE,这种也可以匹配上(r&q ...