一、派生类

在C++编程中,我们在编写一个基类的派生类时,大致可以分为四步

• 吸收基类的成员:不论是数据成员还是函数成员,派生类吸收除基类的构造函数和析构函数之外的全部成员。

• 改造基类函数:在派生类中声明一个或多个与其(某个)基类中的成员函数同名的成员函数,并将它(们)根据新的需求进行重写

• 发展新的成员:在派生类中添加新的成员变量和成员函数,其中新添加的成员要求必须和基类中的成员不同名,并且应当保证新添加的成员会使派生类在功能上相比其基类有所发展

• 重写派生类的构造函数和析构函数

特别注意:

在重写派生类的构造函数时,通常将其基类们的构造函数变成新构造函数的一部分

语法:派生类名::派生类名(参数总表):基类名1(参数表1),基类名2(参数表2),......,新增成员名1(参数1),新增成员名2(参数2){}

示例:

 #include<iostream>
using namespace std;
3 class Father{ //基类
4 public:
5 Father()=default; //默认构造函数
6 Father(int value):father_value(value){} //带参数的构造函数
7 ~Father(){} //析构函数
8 void show(){
9 cout<<"这是基类Father"<<endl;
10 }
11 void father_func(){
12 cout<<"这是基类Father的方法father_func:"<<endl;
13 cout<<"\t成员father_value的值为:"<<father_value<<endl;
14 }
15 private:
16 int father_value;
17 };

19 class Son:public Father{ //Father类的派生类Son
20 public:
/*步骤一:吸收Father类中除构造函数和析构函数之外的所有成员
22 void show(){
23 cout<<"这是基类Father"<<endl;
24 }
25 void father_func(){
26 cout<<"这是基类Father的方法father_func:"<<endl;
27 cout<<"\t成员father_value的值为:"<<father_value<<endl;
28 }
29 */
30 //步骤二:改造基类Father中的成员
31 void show(){
32 cout<<"这是派生类Son"<<endl;
33 }
//步骤三:发展新的成员
35 void son_func(){
36 cout<<"这是派生类Son的方法son_func: "<<endl;
37 cout<<"\t成员son_value的值为:"<<son_value<<endl;
38 }
//步骤四:重写构造函数和析构函数
40 Son()=default;
41 Son(int value):Father(value),son_value(value){}
42 ~Son(){}
43 private:
44 /*步骤一:吸收Father类中除构造函数和析构函数之外的所有成员
int father_value;
*/
//步骤三:发展新的成员
48 int son_value;
49 };
int main(){
Father father();
father.show();
father.father_func();
cout<<"------------分界线--------------------"<<endl;
Son son();
son.show();
son.father_func();
son.son_func();
return ;
}

二、派生类的构造函数的调用顺序

我们先来看一个示例:

 #include<iostream>
using namespace std;
class Father1{ //基类1
public:
Father1(){
cout<<"这是Father1类的构造函数"<<endl;
}
}; class Father2{ //基类2
public:
Father2(){
cout<<"这是Father2类的构造函数"<<endl;
}
}; class Father3{ //基类3
public:
Father3(){
cout<<"这是Father3类的构造函数"<<endl;
}
}; class Son:public Father1,public Father2,public Father3{ //派生类
public:
Son(){
cout<<"这是Son类的构造函数"<<endl;
}
}; int main(){
Son s;
return ;
}

由上面的例子可以看出,派生类在创建对象时会先调用其基类们的构造函数,然后才会调用自己的构造函数。下面是类Son的对象s在内存中的存放形式:

那么派生类调用基类的构造函数的顺序又是如何确定的呢?我们在来看一个例子:

 #include<iostream>
using namespace std;
class Father1{ //基类1
public:
Father1(){
cout<<"这是Father1类的构造函数"<<endl;
}
}; class Father2{ //基类2
public:
Father2(){
cout<<"这是Father2类的构造函数"<<endl;
}
}; class Father3{ //基类3
public:
Father3(){
cout<<"这是Father3类的构造函数"<<endl;
}
}; class Son:public Father3,public Father1,public Father2{ //派生类
public:
Son(){
cout<<"这是Son类的构造函数"<<endl;
}
private:
30 Father1 father1;
31 Father2 father2;
32 Father3 father3;
}; int main(){
Son s;
return ;
}

由此可见,派生类在创建对象时其调用构造函数的顺序是:

先按照派生类对基类的继承顺序调用基类的构造函数。上例中由语句“class Son:public Father3,public Father1,public Father2”可知类Son先继承基类Father3,然后继承基类Father1,最后继承基类Father2,因此其调用基类的构造函数的顺序也是先Fathe3,再Father1,最后Father2。

•( 若派生类的成员变量中存在其基类的对象 )接着按照基类的对象在派生类定义中声明的先后顺序调用基类的构造函数。上例中按照成员对象的定义顺序依次调用Father1、Father2和Father3的构造函数。

• 最后调用派生类自己的构造函数

特别注意:

1.如果基类中没有定义默认构造函数或带有缺省值的构造函数而只有带参数的构造函数时,派生类的构造函数中必须显式的给出基类名和参数表,否则编译器将报错

 #include<iostream>
using namespace std;
class Father1{ //基类1
public:
Father1(int v):value1(v){
cout<<"这是Father1类的构造函数"<<endl;
}
private:
int value1;
}; class Father2{ //基类2
public:
Father2(int v):value2(v){
cout<<"这是Father2类的构造函数"<<endl;
}
private:
int value2;
}; class Father3{ //基类3
public:
Father3(int v):value3(v){
cout<<"这是Father3类的构造函数"<<endl;
}
private:
int value3;
}; class Son:public Father1,public Father2,public Father3{ //派生类
public:
Son(int v):Father1(v),Father2(v),Father3(v),value4(v){ //派生类的构造函数中必须显式的给出基类名和参数表
cout<<"这是Son类的构造函数"<<endl;
}
private:
int value4;
}; int main(){
Son s();
return ;
}

2.如果基类中没有定义构造函数,这派生类也可以不定义构造函数,系统会自动在类中添加默认的构造函数的

3.如果基类中定义了带有参数表的构造函数时,派生类就应当定义相应的构造函数

QUESTION:基类在派生类的构造函数的初始化列表中的顺序是否会影响派生类的构造函数调用顺序?

ANSWER: 我们先来看一个示例:

 #include<iostream>
using namespace std;
class Father1{ //基类1
public:
Father1(int v):value1(v){
cout<<"这是Father1类的构造函数"<<endl;
}
private:
int value1;
}; class Father2{ //基类2
public:
Father2(int v):value2(v){
cout<<"这是Father2类的构造函数"<<endl;
}
private:
int value2;
}; class Father3{ //基类3
public:
Father3(int v):value3(v){
cout<<"这是Father3类的构造函数"<<endl;
}
private:
int value3;
}; class Son:public Father3,public Father1,public Father2{ //派生类
public:
Son(int v):Father1(v),Father2(v),Father3(v),value4(v){
cout<<"这是Son类的构造函数"<<endl;
}
private:
int value4;
}; int main(){
Son s();
return ;
}

上面的例子中派生类Son是以顺序Father3、Father1、Father2来继承基类的,但在其构造函数中基类的顺序确实Father1、Father2、Father3。最终的结果表明类Son仍以其继承基类的顺序来调用基类的构造函数,而非基类在派生类构造函数中的顺序。因此可见基类在派生类的构造函数的初始化列表中的顺序不会影响派生类的构造函数调用顺序

三、派生类的析构函数的调用顺序

• 在派生类中,其析构函数只需要关心新增的一般成员的“善后工作”。而对于新增的成员对象和基类的“善后工作”,系统会自己调用成员对象和基类的析构函数来完成,而不需要用户来关心。

• 在派生类中,析构函数各部分的执行顺序与其构造函数的调用顺序刚好相反,即派生类的析构函数先对其新增的一般成员进行析构,然后对新增的成员对象进行析构,最后按照与其继承基类的相反顺序来调用基类的析构函数。

 #include<iostream>
using namespace std;
class Father1{ //基类1
public:
Father1(){
cout<<"这是Father1类的构造函数"<<endl;
}
~Father1(){
cout<<"这是Father1类的析构函数"<<endl;
}
}; class Father2{ //基类2
public:
Father2(){
cout<<"这是Father2类的构造函数"<<endl;
}
~Father2(){
cout<<"这是Father2类的析构函数"<<endl;
}
}; class Father3{ //基类3
public:
Father3(){
cout<<"这是Father3类的构造函数"<<endl;
}
~Father3(){
cout<<"这是Father3类的析构函数"<<endl;
}
}; class Son:public Father3,public Father1,public Father2{ //派生类
public:
Son(){
cout<<"这是Son类的构造函数"<<endl;
}
~Son(){
cout<<"这是Son类的析构函数"<<endl;
}
private:
Father1 father1;
Father2 father2;
Father3 father3;
}; int main(){
Son s;
return ;
}

C++:派生类的构造函数和析构函数的调用顺序的更多相关文章

  1. c++学习笔记4,派生类的构造函数与析构函数的调用顺序(一)

    測试源代码: //測试派生类的构造函数的调用顺序何时调用 //Fedora20 gcc version=4.8.2 #include <iostream> using namespace ...

  2. C++学习笔记(6)----基类和派生类的构造函数和析构函数的执行顺序

    基类和派生类:构造函数和析构函数的执行顺序 在Visual Studio中,新建控制台工程,构造类如下: #include<iostream> using namespace std; c ...

  3. C++:派生类的构造函数和析构函数

    4.2 派生类的构造函数和析构函数4.2.1 派生类构造函数和析构函数的执行顺序 通常情况下,当创建派生类对象时,首先执行基类的构造函数,随后再执行派生类的构造函数:当撤销派生类对象时,则先执行派生类 ...

  4. 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成员)

    [源码下载] 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成 ...

  5. C++学习之路—继承与派生(二):派生类的构造函数与析构函数

    (根据<C++程序设计>(谭浩强)整理,整理者:华科小涛,@http://www.cnblogs.com/hust-ghtao转载请注明) 由于基类的构造函数和析构函数是不能被继承的,所以 ...

  6. cc28c_demo.cpp,派生类的构造函数和析构函数-代码示范3

    cc28c_demo.cpp,派生类的构造函数和析构函数-代码示范3 //派生类的构造函数和析构函数//派生类的构造函数(执行步骤)//--执行基类的构造函数//--执行成员对象的构造函数//--执行 ...

  7. C++学习笔记(7)----类的数组中构造函数和析构函数的调用顺序

    C++类的数组中构造函数和析构函数的调用顺序(2) 对于如下的代码: #include<iostream> using namespace std; class CBase { priva ...

  8. C++C++中构造函数与析构函数的调用顺序

    http://blog.csdn.net/xw13106209/article/details/6899370 1.参考文献 参考1: C++继承中构造函数.析构函数调用顺序及虚函数的动态绑定 参考2 ...

  9. C++ 构造函数和析构函数的调用顺序、虚析构函数的作用

    构造函数和析构函数的调用顺序 构造函数的调用顺序: 当建立一个对象时,首先调用基类的构造函数,然后调用下一个派生类的构造函数,依次类推,直至到达最底层的目标派生类的构造函数为止. 析构函数的调用书序: ...

随机推荐

  1. Git提交分支

    Git提交分支操作 1.git add 命令告诉 Git 开始对这些文件进行跟踪 git add . 2.然后提交 git commit -m'这是注释信息' 3.git pull命令用于从另一个存储 ...

  2. docker tomcat 已主机名为日志输出路径

    目的:所有的日志输出到共享存储目录中 方法:将 tomcat 的日志放置到 /data/logs/主机名/  下, 1. 修改tomcat/conf下的logging.properties [root ...

  3. 5、JUC--实现 Callable 接口

    Callable接口  Java 5.0 在 java.util.concurrent 提供了一个新的创建执行 线程的方式:Callable 接口  Callable 接口类似于 Runnable ...

  4. 基于DirectX的半球形天空类的C++和C#实现

    目前,天空绘制主要有三种方法:矩形天空.天空盒和球形天空. (1)矩形天空使用一个与地面垂直或呈一定夹角的矩形表示天空,用接近于天空的颜色或云彩纹理贴于矩形上.这种方法简单易行,但需要不断调整视角或观 ...

  5. 浅拷贝与深拷贝的实现方式、区别;deepcopy如果你来设计,如何实现(一)

    浅拷贝与深拷贝的实现方式.区别:deepcopy如果你来设计,如何实现: copy浅拷贝:没有拷贝子对象,所以原始数据改变,子对象改变 deepcopy深拷贝:包含对象里面的子对象的拷贝,所以原始对象 ...

  6. poj 3169 Layout(线性差分约束,spfa:跑最短路+判断负环)

    Layout Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 15349   Accepted: 7379 Descripti ...

  7. 写个定时任务更新svn

    最近学了点shell编程,寻思锻炼下写一个.平常你学习或者看别人讲,自己不练习肯定不行,基本上一动手准出错哈哈.等自己去实践,才会知道哪里有问题,哪里容易出错,哪里要注意什么的. 因为我们每个人有自己 ...

  8. 在main函数前后执行的函数之 C语言

    在gcc中,可以使用attribute关键字,声明constructor和destructor,来指定了函数在main之前或之后运行,代码如下: #include <stdio.h> __ ...

  9. CAN总线学习系列之— CAN总线特点介绍

    CAN总线学习系列之— CAN总线特点介绍 CAN 总线作为一种工业界的流行总线广泛应于工业自动化.多种控制设备.交通工具.医疗仪器以及建筑.环境控制等各个行业中,它是是一种多主机局域网,所以这样 一 ...

  10. Transaction Check Error:file /usr/libexec/getconf/default conflicts between attempted installs of gcc-6.4.1-1.fc25.i686 and gcc-6.4.1-1.fc25.x86_64

    今天在我的ubuntu系统上使用yum来安装软件时出错了错误:Transaction Check Error:file /usr/libexec/getconf/default conflicts b ...