C++ 子类父类构造函数调用顺序

下面是一个a.cpp
 
#include <stdio.h>
class A
{
    public:
        A(){puts("A\n");}
};


class B:public A
{
    public:
        B(){puts("B\n");}
};


int main()
{
    B b;//

    printf ("-------------------\n");
    
    B c();// 不会调用构造函数,这相当于声明了一个函数 类似 int f(); 
    return 0;
}

 
执行结果:
root@anybackup:~/sxz# ./a
A

B

-------------------

调用顺序,下面来看汇编代码



root@anybackup:~/sxz# g++ -S a.cpp 
root@anybackup:~/sxz# cat a.s
.file "a.cpp"
.section .rodata
.LC0:
.string "A\n"
.section .text._ZN1AC2Ev,"axG",@progbits,_ZN1AC5Ev,comdat
.align 2
.weak _ZN1AC2Ev
.type _ZN1AC2Ev, @function
_ZN1AC2Ev:   //A的构造函数
.LFB1:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
subq $16, %rsp
movq %rdi, -8(%rbp)
movl $.LC0, %edi
call puts   //输出 A
leave
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE1:
.size _ZN1AC2Ev, .-_ZN1AC2Ev
.weak _ZN1AC1Ev
.set _ZN1AC1Ev,_ZN1AC2Ev
.section .rodata
.LC1:
.string "B\n"
.section .text._ZN1BC2Ev,"axG",@progbits,_ZN1BC5Ev,comdat
.align 2
.weak _ZN1BC2Ev
.type _ZN1BC2Ev, @function
_ZN1BC2Ev:  //B的构造函数
.LFB4:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
subq $16, %rsp
movq %rdi, -8(%rbp)
movq -8(%rbp), %rax
movq %rax, %rdi
call _ZN1AC2Ev //调用A的构造函数
movl $.LC1, %edi
call puts  //输出 B
leave
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE4:
.size _ZN1BC2Ev, .-_ZN1BC2Ev
.weak _ZN1BC1Ev
.set _ZN1BC1Ev,_ZN1BC2Ev
.section .rodata
.LC2:
.string "-------------------"
.text
.globl main
.type main, @function
main :
.LFB6:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
subq $16, %rsp
leaq -1(%rbp), %rax
movq %rax, %rdi
call _ZN1BC1Ev //调用B的构造函数
movl $.LC2, %edi
call puts
movl $0, %eax
leave
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE6:
.size main, .-main
.ident "GCC: (Ubuntu 4.8.2-19ubuntu1) 4.8.2"
.section .note.GNU-stack,"",@progbits

可以看出是在B的构造函数里调用的A的构造函数,而且是隐式调用的






转载于:https://my.oschina.net/u/1011432/blog/389367

### 回答1: 在Java中,子类构造函数在创建对象时会默认调用父类的无参构造函数。如果父类中没有无参构造函数,那么子类构造函数必须在第一行显式的调用父类的有参构造函数子类构造函数调用父类构造函数的方式有两种: 1. 使用super关键字调用父类构造函数 使用super关键字调用父类构造函数时,需要在子类构造函数的第一行使用super关键字,并传递相应的参数,如: ``` public class Son extends Father { public Son(String name, int age) { super(name, age); // 子类构造函数其他操作 } } ``` 2. 在父类中定义有参构造函数,并在子类构造函数调用父类中定义有参构造函数时,子类构造函数不必使用super关键字调用父类构造函数,只需在子类构造函数中传入相应的参数即可,如: ``` public class Father { String name; int age; public Father(String name, int age) { this.name = name; this.age = age; } } public class Son extends Father { public Son(String name, int age) { // 子类构造函数中不需要调用super关键字 // 父类中有参构造函数可以直接使用 this.name = name; this.age = age; // 子类构造函数其他操作 } } ``` 以上就是子类构造函数调用父类构造函数的两种方式,具体使用哪种方式要根据实际情况来决定。 ### 回答2: 在面向对象的编程中,子类继承了父类的属性和方法,并且可以在此基础上添加额外的属性和方法。在子类构造函数中,我们需要调用父类构造函数来初始化继承的属性。 子类通过使用super关键字来调用父类构造函数。super关键字既可以用来调用父类构造函数,也可以用来调用父类的方法。调用父类构造函数有两种方式:一是调用无参的构造函数,二是调用有参的构造函数。 如果父类定义了无参构造函数子类可以直接调用它来初始化继承的属性。例如: ``` public class Parent{ private int age; private String name; public Parent(){ age = 20; name = "ZhangSan"; } } public class Child extends Parent{ private String school; public Child(){ super(); // 调用父类的无参构造函数 school = "Tsinghua University"; } } ``` 如果父类定义了有参构造函数,就需要在子类构造函数中显示地调用它。例如: ``` public class Parent{ private int age; private String name; public Parent(int age,String name){ this.age = age; this.name = name; } } public class Child extends Parent{ private String school; public Child(int age,String name,String school){ super(age,name); // 调用父类的有参构造函数 this.school = school; } } ``` 在调用父类构造函数时,需要注意以下几点: 1. super()必须是子类构造函数中的第一行代码; 2. 如果子类调用父类构造函数是有参构造函数,也要确保父类有相应的构造函数; 3. 如果子类没有显示地调用父类构造函数,Java编译器会自动调用父类的无参构造函数。 总之,在子类构造函数调用父类构造函数是初始化继承属性的必要步骤,也是实现类之间继承的核心机制。对于Java的面向对象程序设计,这一点不容忽视。 ### 回答3: 在面向对象编程中,一个类可以作为另一个类的子类(也称为派生类)。子类可以继承父类的属性和方法,并且还可以添加自己的属性和方法。在子类构造函数中,可以通过调用父类构造函数来实现对父类属性的赋值。 在C++中,子类构造函数调用父类构造函数的方式是在子类构造函数初始化列表中调用父类构造函数。如下所示: ``` class Parent{ public: Parent(int a){ this->a = a; } private: int a; }; class Child: public Parent{ public: Child(int a, int b): Parent(a){ //调用父类构造函数 this->b = b; } private: int b; }; ``` 在上述代码中,Child是Parent的子类。当Child的构造函数调用时,会首先调用Parent的构造函数,初始化Parent类中的成员变量a,然后再初始化Child类中的成员变量b。 需要注意的是,在子类构造函数中,必须首先调用父类构造函数,因为子类继承了父类的属性和方法,需要确保这些属性和方法在子类对象被创建时已经得到正确的初始化。 除了初始化列表,还可以在子类构造函数中通过调用父类构造函数的方式来实现对父类属性的赋值,但是这种方式更加繁琐,且容易出错。因此,在子类构造函数中,最好采用初始化列表的方式调用父类构造函数进行赋值。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值