由于派生类包含基类的原因,我们在创建一个派生类的时候,系统会先创建一个基类。需要注意的是,派生类会吸纳基类的全部成员,但并不包括构造函数及后面讲的析构函数,那么就意味着创建派生类在调用自己的构造函数之前,会先调用基类的构造函数。
这里一点我们可以通过代码验证:
#include<iostream>usingnamespacestd;classClock{private:intH;intM;intS;public:Clock(){cout<<Clock'sConstructorCalled!<<endl;}};classAlarmClock:publicClock{private:intAH;intAM;public:AlarmClock(){cout<<AlarmClock'sConstructorCalled!<<endl;}};intmain(){AlarmClockA;return0;}
运行截图如下:
我们可以看到仅仅定义了一个派生类对象,派生类和基类的构造函数会自动调用,调用顺序是先调用基类的构造函数再调用派生类的构造函数。
以上大家看到的是最常见也最简单的调用方法,这仅仅是隐式的,也就是不用写出来的、自动的调用。那么当基类的构造函数是带参数的情况下如何调用呢?这样还可以吗?如何传参呢?
答:那就需要我们显式的,也就是明确的写出来,并指定参数传递,来告诉编译器。
一般的写法格式为:
派生类构造函数名(总形参表列):基类构造函数(实参表列)
例如代码:
#include<iostream>usingnamespacestd;classClock{private:intH;intM;intS;public:Clock(){cout<<Clock'sConstructorCalled!<<endl;}Clock(inth,intm,ints){this->H=h;this->M=m;this->S=s;cout<<Clock'sConstructorwithparameterCalled!<<endl;}};classAlarmClock:publicClock{private:intAH;intAM;public:AlarmClock(){cout<<AlarmClock'sConstructorCalled!<<endl;}AlarmClock(inth,intm,ints):Clock(h,m,s){cout<<AlarmClock'sConstructorwithparameterCalled!<<endl;}};intmain(){AlarmClockA(8,10,30);AlarmClockB;return0;}
大家注意看派生类的构造函数,后面通过冒号跟基类的传参,且基类里的参数里为实参,来实现显示的参数调用。请大家上机实践!
需要注意的是,一旦基类中有带参数的构造函数,派生类中则必须有显式传参的派生类构造函数,来实现基类中参数的传递,完成初始化工作。