C++派生类的构造函数

基类的构造函数不能被继承,在声明派生类时,对继承过来的成员变量的初始化工作也要由派生类的构造函数来完成。所以在设计派生类的构造函数时,不仅要考虑派生类新增的成员变量,还要考虑基类的成员变量,要让它们都被初始化。

解决这个问题的思路是:在执行派生类的构造函数时,调用基类的构造函数。

下面的例子展示了如何在派生类的构造函数中调用基类的构造函数。
  1. #include<iostream>
  2. using namespace std;
  3. //基类
  4. class People{
  5. protected:
  6. char *name;
  7. int age;
  8. public:
  9. People(char*, int);
  10. };
  11. People::People(char *name, int age): name(name), age(age){}
  12. //派生类
  13. class Student: public People{
  14. private:
  15. float score;
  16. public:
  17. Student(char*, int, float);
  18. void display();
  19. };
  20. //调用了基类的构造函数
  21. Student::Student(char *name, int age, float score): People(name, age){
  22. this->score = score;
  23. }
  24. void Student::display(){
  25. cout<<name<<"的年龄是"<<age<<",成绩是"<<score<<endl;
  26. }
  27. int main(){
  28. Student stu("小明", 16, 90.5);
  29. stu.display();
  30. return 0;
  31. }
运行结果为:
小明的年龄是16,成绩是90.5

请注意代码第23行:
Student::Student(char *name, int age, float score): People(name, age)
这是派生类 Student 的构造函数的写法。冒号前面是派生类构造函数的头部,这和我们以前介绍的构造函数的形式一样,但它的形参列表包括了初始化基类和派生类的成员变量所需的数据;冒号后面是对基类构造函数的调用,这和普通构造函数的参数初始化表非常类似。

实际上,你可以将对基类构造函数的调用和参数初始化表放在一起,如下所示:
Student::Student(char *name, int age, float score): People(name, age), score(score){}
基类构造函数和初始化表用逗号隔开。

需要注意的是:冒号后面是对基类构造函数的调用,而不是声明,所以括号里的参数是实参,它们不但可以是派生类构造函数总参数表中的参数,还可以是局部变量、常量等。如下所示:
Student::Student(char *name, int age, float score): People("李磊", 20)

基类构造函数调用规则

事实上,通过派生类创建对象时必须要调用基类的构造函数,这是语法规定。也就是说,定义派生类构造函数时最好指明基类构造函数;如果不指明,就调用基类的默认构造函数(不带参数的构造函数);如果没有默认构造函数,那么编译失败。

请看下面的例子:
  1. #include<iostream>
  2. using namespace std;
  3. //基类
  4. class People{
  5. protected:
  6. char *name;
  7. int age;
  8. public:
  9. People();
  10. People(char*, int);
  11. };
  12. People::People(){
  13. this->name = "xxx";
  14. this->age = 0;
  15. }
  16. People::People(char *name, int age): name(name), age(age){}
  17. //派生类
  18. class Student: public People{
  19. private:
  20. float score;
  21. public:
  22. Student();
  23. Student(char*, int, float);
  24. void display();
  25. };
  26. Student::Student(){
  27. this->score = 0.0;
  28. }
  29. Student::Student(char *name, int age, float score): People(name, age){
  30. this->score = score;
  31. }
  32. void Student::display(){
  33. cout<<name<<"的年龄是"<<age<<",成绩是"<<score<<endl;
  34. }
  35. int main(){
  36. Student stu1;
  37. stu1.display();
  38. Student stu2("小明", 16, 90.5);
  39. stu2.display();
  40. return 0;
  41. }
运行结果:
xxx的年龄是0,成绩是0
小明的年龄是16,成绩是90.5

创建对象 stu1 时,执行派生类的构造函数 Student::Student(),它并没有指明要调用基类的哪一个构造函数,从运行结果可以很明显地看出来,系统默认调用了不带参数的构造函数,也就是 People::People()。

创建对象 stu2 时,执行派生类的构造函数 Student::Student(char *name, int age, float score),它指明了基类的构造函数。

在第31行代码中,如果将 People(name, age) 去掉,也会调用默认构造函数,stu2.display() 的输出结果将变为:
xxx的年龄是0,成绩是90.5

如果将基类 People 中不带参数的构造函数删除,那么会发生编译错误,因为创建对象 stu1 时没有调用基类构造函数。

总结:如果基类有默认构造函数,那么在派生类构造函数中可以不指明,系统会默认调用;如果没有,那么必须要指明,否则系统不知道如何调用基类的构造函数。

构造函数的调用顺序

为了搞清这个问题,我们不妨先来看一个例子:
  1. #include<iostream>
  2. using namespace std;
  3. //基类
  4. class People{
  5. protected:
  6. char *name;
  7. int age;
  8. public:
  9. People();
  10. People(char*, int);
  11. };
  12. People::People(): name("xxx"), age(0){
  13. cout<<"PeoPle::People()"<<endl;
  14. }
  15. People::People(char *name, int age): name(name), age(age){
  16. cout<<"PeoPle::People(char *, int)"<<endl;
  17. }
  18. //派生类
  19. class Student: public People{
  20. private:
  21. float score;
  22. public:
  23. Student();
  24. Student(char*, int, float);
  25. };
  26. Student::Student(): score(0.0){
  27. cout<<"Student::Student()"<<endl;
  28. }
  29. Student::Student(char *name, int age, float score): People(name, age), score(score){
  30. cout<<"Student::Student(char*, int, float)"<<endl;
  31. }
  32. int main(){
  33. Student stu1;
  34. cout<<"--------------------"<<endl;
  35. Student stu2("小明", 16, 90.5);
  36. return 0;
  37. }
运行结果:
PeoPle::People()
Student::Student()
--------------------
PeoPle::People(char *, int)
Student::Student(char*, int, float)

从运行结果可以清楚地看到,当创建派生类对象时,先调用基类构造函数,再调用派生类构造函数。如果继承关系有好几层的话,例如:

A --> B --> C

那么则创建C类对象时,构造函数的执行顺序为:

A类构造函数 --> B类构造函数 --> C类构造函数

构造函数的调用顺序是按照继承的层次自顶向下、从基类再到派生类的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值