class A{
public:
A(){
int a = 0;
};
virtual ~A(){
int a = 1;
};
virtual int test(){
return 1;
};
};
class B:public A{
public:
B(){
int a = 2;
};
virtual ~B(){
int a = 3;
};
virtual int test(){
return 2;
};
};
执行以下操作:
A* p = new B;
delete p;
p->test();
根据我以前知道的知识,在new子类时,会先调用基类构造,再调用子类构造;在delete子类时,会先调用子类析构,再调用基类析构;
而在调用子类的虚函数test时,并不会在调用前或调用后自动调用基类的test函数。
首先纠正一个错误:在执行new时我们可以观察到,程序会先执行基类A构造函数中的操作,再执行子类B构造函数中的操作,但这不代表先调用了A的构造,其后才调用的B的构造。
如果仔细观察汇编代码,可以发现main确确实实的call了 B::B()
mian():
push rbp
mov rbp, rsp
push rbx
sub rsp, 24
mov edi, 8
call operator new(unsigned long)
mov rbx, rax
mov rdi, rbx
call B::B() [complete object constructor]
mov QWORD PTR [rbp-24], rbx
mov rax, QWORD PTR [rbp-24]
test rax, rax
je .L12
mov rdx, QWORD PTR [rax]
add rdx, 8
mov rdx, QWORD PTR [rdx]
mov rdi, rax
call rdx
其实这个问题原因出在编译器在编译存在继承关系的类的构造与析构时的特殊对待:
在编译B的构造函数时,在处理完栈指针和参数后(参数可能是B对象的地址),会调用A的构造,此后再执行B构造中的操作。
B::B() [base object constructor]:
push rbp
mov rbp, rsp
sub rsp, 32
mov QWORD PTR [rbp-24], rdi
mov rax, QWORD PTR [rbp-24]
mov rdi, rax
call A::A() [base object constructor]
mov edx, OFFSET FLAT:vtable for B+16
mov rax, QWORD PTR [rbp-24]
mov QWORD PTR [rax], rdx
mov DWORD PTR [rbp-4], 2
nop
leave
ret
同样的,在编译B的析构函数时,在执行完整个函数准备ret前,会调用A的析构函数。
B::~B() [base object destructor]:
push rbp
mov rbp, rsp
sub rsp, 32
mov QWORD PTR [rbp-24], rdi
mov edx, OFFSET FLAT:vtable for B+16
mov rax, QWORD PTR [rbp-24]
mov QWORD PTR [rax], rdx
mov DWORD PTR [rbp-4], 3
mov rax, QWORD PTR [rbp-24]
mov rdi, rax
call A::~A() [base object destructor]
nop
leave
ret
B::~B() [deleting destructor]:
push rbp
mov rbp, rsp
sub rsp, 16
mov QWORD PTR [rbp-8], rdi
mov rax, QWORD PTR [rbp-8]
mov rdi, rax
call B::~B() [complete object destructor]
mov rax, QWORD PTR [rbp-8]
mov esi, 8
mov rdi, rax
call operator delete(void*, unsigned long)
leave
ret
子类的构造与析构如此特殊在于编译器对它们的特殊处理,子类的构造函数在执行第一行语句前,会调用基类的构造函数;而子类的析构函数在执行完所有操作准备ret前,会调用基类的析构函数。
这也就导致我们能够观察到多重继承中的层层构造和层层析构,因为父类的构造(析构)已经被编译到子类的构造(析构)中,自然会有这样的调用顺序!甚至就算在子类的析构中插入return语句,都只会jmp到调用父类析构前的语句,而不是像普通函数一样直接ret。