学无止尽,积土成山,积水成渊-《C++反汇编与逆向分析技术揭秘》 读书笔记

对象的内存布局

一般计算公式:

对象内存大小 = sizeof(数据成员1)+ sizeof(数据成员2) +. .. + sizeof(数据成员n)
 
若类中没有继承和虚函数的定义,还有三种特殊情况考虑:空类、内存对齐、静态成员函数。
  • 空类:空类的长度为1字节,如果不占用字节的话,this指针会悬空。考虑到类可以仅有成员函数,没有数据成员。
  • 内存对齐:在VC中,类和结构体中的数据成员是根据在类或结构中出现的顺序来依次申请空间的,由于内存对齐原因,可能不会连续的排列,数据成员之间可能有间隙。
  • 静态变量:与静态全局变量类似,存在的位置和全局变量一致,只是编译器增加了作用域的检查,作用域之外不可见。

访问对象中的数据成员时,一般采用寄存器间接相对寻址,表达式esp±n±offset,或ebp±n±offset(其中esp±n或ebp±n是对象的首地址,offset 为数据成员相对对象首地址的偏移)。由于n和offset在编译阶段时属于常量,采用编译器优化时,表达式可能简化为 ebp±n 或 esp±n。

this指针

this指针是保存所属对象的首地址。在调用成员函数时,遵循默认的thiscall约定,利用寄存器ECX保存对象的首地址(即this指针),以寄存器传参的方式传递到成员函数中,即是this指针的约定。成员函数中访问成员数据既是通过this指针间接访问的。
  • thiscall并不属于关键字,是C++成员函数特有的调用方式。
  • thiscall的参数压栈顺序也是从右至左
  • thiscall的栈平衡方式与_stdcall相同,由被调用方平衡。
  • 并不是所有的this指针的传递都是通过寄存器ECX,可以强制改用其他调用方式(_如stdcall)
     class CTest
    
          {
    
          public:
    
               void __stdcall SetNumber(int nNumber){
    
                             m_nInt = nNumber;
    
                     }
    
               int nNumber;
    
          }
    
使用thiscall调用方式的成员函数的要点分析
函数调用:
 lea ecx, [mem]            ;取对象地址

 call FUN_ADDRESS          ;调用成员函数

      

函数内部:

 mov XXX, ecx                   ;发现函数内使用ecx中的数据,证明确是通过ecx来传递参数

 mov [reg + ], XXX
 

静态数据成员

静态数据成员和静态变量的原理相同,因此静态数据成员的初值会被写入编译链接后的执行文件中。当程序被加载时,操作系统将执行文件中的数据读到对应的内存单元中,静态数据成员便已经存在,而这时类并没有实例对象。静态数据成员不属于某一个对象,与对象之间是多对一的关系。静态数据成员仅仅和类相关,和对象无关。
所以在计算某个类的对象所占用内存的大小,静态数据成员是不计算在内的。
 
区别:
  • 静态数据成员是常量地址,而普通数据成员一般存储在栈空间。
  • 静态成员通过立即数间接寻址访问,而普通数据成员一般通过寄存器相对间接寻址访问。
  • 静态成员访问时不需要this指针,而普通数据成员访问时需要使用this指针。

对象作为函数参数

 
对象作为函数参数,编译器会把对象视为由几个基本类型的数据的组合,和多个参数函数传参类似。类对象中的数据成员的传参顺序为:最先定义的数据成员最后压栈,最后定义的数据成员最先压栈。当类有构造函数和析构函数,过程会更复杂一些。由于对象在向函数传递过程中,由于复制了对象,等同于又定义了一个对象,会调用复制构造函数。在函数退出时,复制的对象作为函数内部的局部变量被销毁,会调用析构函数。
 
对象作为返回值
 
对象作为函数返回值,与基本类型不同。基本数据类型(双精度浮点数以及非标准的_int64类型除外)作为返回值时,通过寄存器EAX传递。对象作为返回值时,首先在调用函数中申请返回对象使用的栈空间,然后将返回对象的首地址作为参数,通过寄存器EAX传递给被调用函数。在退出被调用函数时,将返回对象中的数据复制到调用函数开辟的返回对象的栈空间,把返回对象的首地址通过寄存器EAX返回。返回的对象是临时存在的,也就是C++临时对象,作用域仅仅限于单条语句。
 
 1 class CReturn{

 2 public:

 3 int m_nNumber;

 4 in m_nArry[];

 5 };

 6 CReturn GetCReturn()

 7 {

 8 CReturn RetObj;

 9 RetObj.m_nNumber = ;

 for(int i=; i< ; i++)

 {

 RetObj.m_nArry[i] = i + ;

 }

 return RetObj;

 }

 void main(int argc, char *argv[])

 {

 CReturn objA;

 objA = GetCReturn();

 printf("%d %d %d", objA.m_nNumber, objA.m_nArry[],  objA.m_nArry[]);

 }
 

C++反汇编-结构体和类的更多相关文章

  1. (五)羽夏看C语言——结构体与类

    写在前面   由于此系列是本人一个字一个字码出来的,包括示例和实验截图.本人非计算机专业,可能对本教程涉及的事物没有了解的足够深入,如有错误,欢迎批评指正. 如有好的建议,欢迎反馈.码字不易,如果本篇 ...

  2. C#中结构体和类的区别

    结构体和类同样能够定义字段,方法和构造函数,都能实例化对象,这样看来结构体和类的功能好像是一样的了,但是他们在数据的存储上是不一样的 C#结构体和类的区别问题:这两种数据类型的本质区别主要是各自指向的 ...

  3. Swift结构体与类

    在面向过程的编程语言(如C语言)中,结构体用得比较多,但是面向对象之后,如在C++和Objective-C中,结构体已经很少使用了.这是因为结构体能够做的事情,类完全可以取而代之.而Swift语言却非 ...

  4. C C++ 中结构体与类

    先来说说C和C++中结构体的不同 a) C语言中的结构体不能为空,否则会报错 1>d:\myproject\visual studio 2013\projects\myc++\main.c(71 ...

  5. 浅析C#中的结构体和类

    类和结构是 .NET Framework 中的常规类型系统的两种基本构造. 两者在本质上都属于数据结构.封装着一组总体作为一个逻辑单位的数据和行为. 数据和行为是该类或结构的"成员" ...

  6. 10 结构体和类 - —— 《Swift3.0 从入门到出家》

    Swift中的面向对象5个要素:枚举.结构体.类.协议.扩展 面向对象研究的是对象,完成一件事情需要多个对象参与,是生活的映射 Swift中结构体和类非常相似,也就是结构体能完成类的所有功能.结构体是 ...

  7. C++中 结构体和类的异同

    在C++中,结构体是一种特殊形态的类. 结构体和类的唯一区别就是:  结构体和类具有不同的默认访问控制属性. 类中,对于未指定访问控制属性的成员,其访问控制属性为私有类型(private) 结构体中, ...

  8. 介绍C#结构体与类区别

    1. 结构体与类定义方式 结构体定义使用struct类定义使用class 结构体: struct testDemo{ int num; void action(){ } } 类: class test ...

  9. C#中结构体与类的区别

    一.结构体和类非常相似 1,定义和使用非常相似,例子如下:public struct Student{    string Name;    int Age;}public class Questio ...

随机推荐

  1. docker stack 部署 mysql 5.6

    =============================================== 2018/7/1_第1次修改                       ccb_warlock === ...

  2. python logging 日志

    logging与print 区别,为什么需要logging? 在写脚本的过程中,为了调试程序,我们往往会写很多print打印输出以便用于验证,验证正确后往往会注释掉,一旦验证的地方比较多,再一一注释比 ...

  3. No.10 selenium学习之路之通过元素定位获取属性

    1. implicitly_wait()隐形等待.等待页面加载完成,作用是全局的. 时间可以设置的长,短时间也没有影响.直到设置的时间耗完 时间耗完也不会报错 2.获取title值 driver.ti ...

  4. Java多线程-Java多线程概述

    第一章 Java多线程概述 线程的启动 线程的暂停 线程的优先级 线程安全相关问题 1.1 进程与线程 进程:可以将运行在内存中的程序(如exe文件)理解为进程,进程是受操作系统管理的基本的运行单元. ...

  5. 利用sys.dm_db_index_physical_stats查看索引大小/碎片等信息

    我们都知道,提高sql server的数据查询速度,最有效的方法,就是为表创建索引,而我们对数据表进行新增,删除,修改的时候,会产生索引碎片,索引碎片多了,对性能产生很大的影响,索引碎片越多对数据库查 ...

  6. MySQL5.6 Waiting for Commit Lock

    mysql  Bug#19843808 自动修复记录 MySQL5.6和Xtrabackup之间存在一个bug,这个bug在5.6.23中已经修复. Xtrabackup备份的时候执行flushs t ...

  7. 微信接口问题(The underlying connection was closed: An unexpected error occurred on a send)

    突然在调用微信接口是报:The underlying connection was closed: An unexpected error occurred on a send错误,跟踪了半天,是因为 ...

  8. Cannot read property ‘msie’ of undefined错误原因以及解决方案

    最近把一个项目的jQuery升级到最新版,发现有些页面报错Cannot read property ‘msie’ of undefined.上jQuery网站上搜了一下,$.browser这个api从 ...

  9. 在Chrome浏览器中保存的密码有多安全?

    本文由 伯乐在线 - 黄利民 翻译.未经许可,禁止转载!英文出处:howtogeek.欢迎加入翻译组. [2013-08-09 更新]:最近又开始讨论“Chrome浏览器明文保存密码这个话题了,国外一 ...

  10. 非ROOT用户不能识别声卡问题

    将非ROOT用户加入到audio组中即可 sudo usermod -a -G audio usrname