一般定义类的时候,我们都会定义构造函数和析构函数,如果我们也想继承这些类,就得知道基类和派生类的构造函数的调用顺序,以及析构函数的调用顺序。(以public继承方式为例)
一、构造函数的调用顺序
我们先来定义两个类,Child类通过public继承Parent类。
class Parent { public: Parent(int a, int b); ~Parent(); void printP(); private: int m_a; int m_b; }; class Child : public Parent { public: Child( int a, int b, int c, int d); ~Child(); void printC(); private: int m_c; int m_d; int m_aa; int m_bb; };
Parent类里面有两个私有成员变量,m_a, m_b, 还有三个共有的成员函数,分别是构造函数,析构函数,还有一个打印函数。
Child类通过public方式继承了Parent里面的成员,但是没有继承基类的构造函数和析构函数。除了继承的基类的成员还有它自己的一些私有成员变量。通过上一篇博客我们知道通过public继承之后的基类的所有属性保持不变。
我们先声明一个Child的对象。
Child c(1,2,3,4);
因为我们在声明基类的时候基类定义了一个有两个参数的构造函数,所以我们必须在声明的时候给基类传两个参数,由于我们Child又是继承Parent类的,所以我们在定义Child类的构造函数时,需要调用Parent类的构造函数。因此我们需要这样来定义子类的构造函数。
Child::Child( int a, int b, int c, int d):Parent(a, b) { cout << "派生类的构造函数" << endl; m_c = c; m_d = d; }
“:”后面的Parent(a, b),就是我们来显示的调用Parent类的构造函数,然后用a,和b分别来初始化Parent类的的私有成员变量m_a, m_b。
结论:
定义一个派生类的对象,首先会调用基类的构造函数,然后再调用派生类的构造函数。
二、析构函数的调用顺序
当函数的结束之后,会自动调用类的析构函数。在继承中的析构函数的调用顺序如下。
先调用派生类的构析构函数,再调用基类的构造函数。
我们可以执行下面的代码验证下:
#include <iostream> using namespace std; class Parent { public: Parent(int a, int b); ~Parent(); void printP(); int geta(); int getb(); private: int m_a; int m_b; }; class Child : public Parent { public: Child( int a, int b, int c, int d); ~Child(); void printC(); private: int m_c; int m_d; int m_aa; int m_bb; }; int Parent::geta() { return m_a; } int Parent::getb() { return m_b; } Parent::Parent(int a, int b) { cout << "基类的构造函数" << endl; m_a = a; m_b = b; } Parent::~Parent() { cout << "基类的析构函数" << endl; } void Parent::printP() { cout << "m_a = " << m_a << endl; cout << "m_b = " << m_b << endl; } Child::Child( int a, int b, int c, int d):Parent(a, b) { cout << "派生类的构造函数" << endl; m_aa = geta(); m_bb = getb(); m_c = c; m_d = d; } Child::~Child() { cout << "派生类的析构函数" << endl; } void Child::printC() { cout << "m_a = " << m_aa << endl; cout << "m_b = " << m_bb << endl; cout << "m_c = " << m_c << endl; cout << "m_d = " << m_d << endl; } int main() { Child c(3,4,5,6); c.printC(); return 0; }
这段代码执行的结果如下。