美文网首页
类实例的构造

类实例的构造

作者: fooboo | 来源:发表于2016-04-28 22:06 被阅读44次

深度探索对象模型【好书】上有一节讲的是什么时候编译器会合成一个构造函数,上面列出了四种情况。因为初始化类的成员变量是程序员的责任,所以不要指望编译器合成的构造函数会做这些工作。在其他情况下,合成出来的构造函数也没什么用。就写两个栗子并反汇编简单看下其中一种情况:

3 class CStudent

4 {

5 public:

6    void ShowInfo()

7    {

8        //TODO

9    }

10 private:

11    int m_iNumber;

12    char m_cSex;

13    int m_iGrade;

14 };

16 int main()

17 {

18    CStudent stu[5];

19    return 0;

20 }

如果我们在main中定义 CStudent stu[5],则反汇编出来的结果是:

804857d:                 push  %ebp

804857e:                 mov    %esp,%ebp

8048580:                 sub    $0x40,%esp

8048583:                 mov    $0x0,%eax

esp桢指针往低地址减了64个字节,因为对齐的原因,一个CStudent实例会占用12个字节,加上预留的。然后什么也没用做。

再看下加了一个虚函数后的代码和汇编:

3 class CStudent

4 {

5 public:

6    virtual void DoNothing()

7    {

8        //TODO

9    }

10

11    void ShowInfo()

12    {

13        //TODO

14    }

15 private:

16    int m_iNumber;

17    char m_cSex;

18    int m_iGrade;

19 };

20

21 int main()

22 {

23    CStudent stu[5];

24    return 0;

25 }

80485e5:  sub    $0x60,%esp               //开辟空间

80485e8:  lea    0x10(%esp),%eax      //取stu首地址,stu[0]在低地址

80485ec:  mov    $0x4,%esi                //调用5次

80485f1:  mov    %eax,%ebx              //stu首地址

80485f3:  jmp    8048603                     //跳转8048603

80485f5:  mov    %ebx,(%esp)            // esp存储stu首地址

80485f8:  call  8048676 <_ZN8CStudentC1Ev>

80485fd:  add    $0x10,%ebx         //对stu + 16 = stu[1],以此类推

8048600:  sub    $0x1,%esi           // esi = 3,2,1,0 -1

8048603:  cmp    $0xffffffff,%esi  //-1 == esi?

8048606:  jne    80485f5  //不等于跳转到80485f5

8048608:  mov    $0x0,%eax

804860d:  lea    -0x8(%ebp),%esp

08048676 <_ZN8CStudentC1Ev>:

8048676: push  %ebp

8048677: mov    %esp,%ebp

8048679: mov    0x8(%ebp),%eax 

804867c: movl  $0x8048728,(%eax)

8048682: pop    %ebp

8048683: ret

现在sizeof(CStudeng) = 16B了,多了个指针成员。其中ZN8CStudentC1Ev是编译器为我们合成的默认构造函数,可以使用c++flit查看,

mov    0x8(%ebp),%eax,为什么是加8,加8取到对象的首地址,因为调用函数需要压栈返回地址和ebp。只做了给每个对象的前四个字节设置$0x8048728,这个是指向虚函数表的值了,为什么是它?他表示什么?这个会在下下一篇中解释到。编译器合成的仅仅设置了这个成员,其他几个数据成员并没有初始化【如果有定义构造函数的话,就不会合成了,而是会扩张每个,在user code前安插编译器的代码,千万不能调用memset等这样清数据的函数,会清了vptr值】。

这就是编译器做了它认为该做的了。像那些含有类实例的类实例,前者有默认构造函数而后者没有,也会合成等等。

下篇打算介绍重载带有默认形参的虚函数的一些不同之处。


相关文章

  • C# 构造函数总结

    构造函数 构造函数分为:实例构造函数,静态构造函数,私有构造函数。 实例构造函数 1、构造函数的名字与类名相同。 ...

  • Java 构造方法中的多态

    类的实例化顺序是先实例化父类在实例化子类,也就是先调用父类的构造方法,再调用子类的构造方法,如果在构造方法中调用其...

  • Swift面向对象

    创建一个类 类的四种属性 类的构造 swift类构造过程 指定构造器 便利构造器 KVC构造器 实例方法和类方法 ...

  • 类实例的构造

    深度探索对象模型【好书】上有一节讲的是什么时候编译器会合成一个构造函数,上面列出了四种情况。因为初始化类的成员变量...

  • C++学习心得

    私有构造函数类的特点 不能实例化,因为实例化时类外部无法访问类的私有构造函数; 不能被继承,因为派生类无法调用类的...

  • Java反射初始化类调用类中的方法

    1.通过无参构造方法实例化对象 无参构造方法,类 反射调用方式 2. 有参构造函数,对象实例化 有参构造函数类 反...

  • 构造函数学习

    1.构造函数是用于初始化类实例的状态。 2.实例化构造函数在创建类的没一个类的实例时执行。 3.如果希望从类的外部...

  • Java Reflection

    类 类的实例对象 如何创建类的实例对象? 直接new通过class实例化(需要有无参数的构造方法) CLass类的...

  • C#继承与构造函数的调用

    1、实例化父类时,可以使用new子类,执行构造函数顺序为:先执行父类构造函数,再执行子类构造函数。 2、实例化子类...

  • 原型与原型链

    1、构造函数 构造函数模式的目的就是为了创建一个自定义类,并且创建这个类的实例。构造函数模式中拥有了类和实例的概念...

网友评论

      本文标题:类实例的构造

      本文链接:https://www.haomeiwen.com/subject/ajqfrttx.html