学无止尽,积土成山,积水成渊-《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. P1183 多边形的面积

    一道睡论数论题 其实是AC300祭才做的水题 题意: 很直白的的题意啊,就是求任意一个多边形的面积 所以我们来安利一下一个求多边形面积的数学通式: 给定多边形的顶点坐标(有序),让你来求这个多边形的面 ...

  2. mybatis 易百练习笔记

    1. session.commit()  增删改需要提交     session.close()    session需要关闭 2. insert  into t()   values()  不用写i ...

  3. R语言学习笔记:使用reshape2包实现整合与重构

    R语言中提供了许多用来整合和重塑数据的强大方法. 整合 aggregate 重塑 reshape 在整合数据时,往往将多组观测值替换为根据这些观测计算的描述统计量. 在重塑数据时,则会通过修改数据的结 ...

  4. Struts 2 - Environment Setup

    Our first task is to get a minimal Struts 2 application running. This chapter will guide you on how ...

  5. C#和PHP 长整型时间互转

    //2018/5/14 16:03:05转换:1526284985 public static double ConvertToDouble(DateTime date) { , , )); var ...

  6. Rookey.Frame之系统初始化

    昨天介绍了数据库的配置,今天继续介绍系统的初始化功能:针对系统初始化在开发中也是很重要的一部分,它可以提前将相关数据提前自动初始化到系统中,同时也可以为上线测试提供方便,可以很方便进行系统测试演练,防 ...

  7. Educational Codeforces Round 9 D - Longest Subsequence

    D - Longest Subsequence 思路:枚举lcm, 每个lcm的答案只能由他的因子获得,类似素数筛搞一下. #include<bits/stdc++.h> #define ...

  8. python之web框架(2):了解WSGI接口

    python之web框架(2):了解WSGI接口 1.什么是wsgi接口: wsgi:Web Service Gateway Interface.它不是模块,而只是一种规范,方便web服务器和各种框架 ...

  9. Git配置用户名密码

    配置Git 在Linux下和windows下配置Git的方法差不多,只是在Linux下,可以在命令行里直接使用git config进行配置, 而在windows下则要先打开“Git Bash”,进入m ...

  10. 请爱护你的JTAG烧录口---记录

        排除了下载线的问题后,还是不能访问FPGA的JTAG口,那么很有可能你的FPGA芯片的JTAG口已经损坏.此时请用万用表检查TCK,TMS,TDO和Tdi是否和GND短路,如果任何一个信号对地 ...