C++继承中的构造和析构

Posted dishengandziyu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++继承中的构造和析构相关的知识,希望对你有一定的参考价值。

1,构造:对象在创建的后所要做的一系列初始化的工作;

      析构:对象在摧毁之前所要做的一系列清理工作;

 

2,思考:

    1,子类中如何初始化父类成员?

       1,对于继承而言,子类可以获得父类的代码,可以获得父类中的成员变量,成员变量从父类中传递到子类,那么子类对象在创建的时候如何初始化从父类中继承的成员变量呢?

    2,父类构造函数和子类构造函数有有什么关系?  

   

3,子类对象的构造:

    1,子类中可以定义构造函数;

         1,子类中定义的新的构造函数,对于子类对象创建的时候必须完成一系列的初始化工作,这些初始化工作包括初始化继承而来的成员,第一种方式直接对他们进行赋值或者直接将他们写到初始化列表中去进行初始化,第二种方式更直接,在子类构造的时候我们调用父类构造函数进行初始化;

         2,这就是子类构造函数可以用来初始化父类成员的两种方式;

    2,子类构造函数:

         1,必须对继承而来的成员进行初始化;

             1,直接通过初始化列表或者赋值的方式进行初始化;

                  1,有可能从父类中继承了 private 成员,因此直接来进行赋值或初始化根本就是行不通的,因此这个时候只能通过父类的构造函数来进行初始化的工作;

             2,调用父类构造函数进行初始化;

    3,父类构造函数在子类中的调用方式;

         1,默认调用:

             1,子类对象在创建的时候会自动的调用父类的构造函数;

             2,适用于无参构造函数和使用默认参数的构造函数;

                  1,对于需要参数的构造函数来说,默认的调用方式是行不通的,必须显示调用;     

         2,显示调用:

             1,通过初始化列表进行调用;

                  1,只能在初始化列表中调用;

             2,适用于所有父类构造函数;

       3,父类构造函数的调用:

           1,代码示例:

 1 class Child : public Parent
 2 {
 3 Public:
 4     Child() /* 隐式调用 */
 5     {
 6          Cout << “Child()” << endl;
 7     }
 8 
 9     Child(string s)  : Parent(“Parameter to Parent”)  /* 显示调用 */ 
10     {
11          Cout << “Child() : “ << s << endl;
12     }
13 };

 

4,子类的构造初探编程实验:

 1 #include <iostream>
 2 #include <string>
 3 
 4 using namespace std;
 5 
 6 class Parent 
 7 {
 8 public:
 9     Parent()
10     {
11         cout << "Parent()" << endl;
12     }
13     
14     Parent(string s)
15     {
16         cout << "Parent(string s) : " << s << endl;
17     }
18 };
19 
20 class Child : public Parent
21 {
22 public:
23     Child()  // 隐式调用父类构造函数中的无参或默认初始值构造函数
24     {
25         cout << "Child()" << endl;
26     }
27     
28     Child(string s)  // 隐式调用父类构造函数中的无参或默认初始值构造函数;
29     {
30         cout << "Child(string s) : " << s << endl;
31     }
32     
33     Child(string s): Parent(s)  // 显示调用父类构造函数;
34     {
35         cout << "Child(string s) : " << s << endl;
36     }
37 };
38 
39 int main()
40 {       
41     Child c;  // 调用父类无参构造函数,然后调用子类无参构造函数;
42     Child cc("cc");  // 调用父类有参构造函数,然后调用子类构造函数;
43     
44     return 0;
45 }

    1,子类和父类的构造函数的调用都严格按照重载规则调用;

   

5,子类对象的构造:

    1,构造规则:

       1,子类对象在创建时会首先调用父类的构造函数(要以子类构造函数调用为依据);

       2,先执行父类构造函数再执行子类构造函数;

       3,父类构造函数可以被隐式调用或者显示调用;

    2,对象创建时构造函数的调用顺序:

       1,调用父类的构造函数;

       2,调用成员变量的构造函数;

       3,调用类自身的构造函数;

       4,口诀:

           1,先父母(调到最上的父母为止),后客人(按照成员变量申明的顺序),再自己;

           2,非常实用;

    3,子类构造深度解析编程实验:

 1 #include <iostream>
 2 #include <string>
 3 
 4 using namespace std;
 5 
 6 class Object
 7 {
 8 public:
 9     Object(string s)
10     {
11         cout << "Object(string s) : " << s << endl;
12     }
13 };
14 
15 class Parent : public Object
16 {
17 public:
18     Parent() : Object("Default")  // 必须显示调用
19     {
20         cout << "Parent()" << endl;
21     }
22     
23     Parent(string s) : Object(s)  // 必须显示调用
24     {
25         cout << "Parent(string s) : " << s << endl;
26     }
27 };
28 
29 class Child : public Parent
30 {
31     Object mO1;  // 组合关系,默认构造函数和有参数构造函数,这里必须显示调用
32     Object mO2;
33     
34 public:
35     Child() : mO1("Default 1"), mO2("Default 2")
36     {
37         cout << "Child()" << endl;
38     }
39     
40     Child(string s) : Parent(s), mO1(s + " 1"), mO2(s + " 2")  // 父母和成员客人的区别在于父母构造函数直接调用,客人构造函数通过对象调用;
41     {
42         cout << "Child(string s) : " << s << endl;
43     }
44 };
45 
46 int main()
47 {       
48     Child cc("cc");
49     
50     // cc output:
51     // Object(string s) : cc  父母
52     // Parent(string s) : cc  父母
53     // Object(string s) : cc 1  客人
54     // Object(string s) : cc 2  客人
55     // Child(string s) :  cc  自己
56     
57     return 0;
58 }

   

6,子类对象的析构:

    1,析构函数的调用顺序与构造函数相反:

       1,执行自身的析构函数;

       2,执行成员变量的析构函数;

       3,执行父类的析构函数;

    2,对象的析构编程实验:

 1 #include <iostream>
 2 #include <string>
 3 
 4 using namespace std;
 5 
 6 class Object
 7 {
 8     string ms;  // 为了验证析构函数的输出,定义了这个成员变量来传递值,因为构造函数可以由外界传入值,但是析构函数没有参数,是不可以由完结传入值的;
 9     
10 public:
11     Object(string s)
12     {
13         cout << "Object(string s) : " << s << endl;
14         ms = s;
15     }
16     
17     ~Object()
18     {
19         cout << "~Object() : " << ms << endl;
20     }
21 };
22 
23 class Parent : public Object
24 {
25     string ms;
26     
27 public:
28     Parent() : Object("Default")
29     {
30         cout << "Parent()" << endl;
31         ms = "Default";
32     }
33     
34     Parent(string s) : Object(s)
35     {
36         cout << "Parent(string s) : " << s << endl;
37         ms = s;
38     }
39     
40     ~Parent()
41     {
42         cout << "~Parent() : " << ms << endl;
43     }
44 };
45 
46 class Child : public Parent
47 {
48     Object mO1;
49     Object mO2;
50     string ms;
51     
52 public:
53     Child() : mO1("Default 1"), mO2("Default 2")
54     {
55         cout << "Child()" << endl;
56         ms = "Default";
57     }
58     
59     Child(string s) : Parent(s), mO1(s + " 1"), mO2(s + " 2")
60     {
61         cout << "Child(string s) : " << s << endl;
62         ms = s;
63     }
64     
65     ~Child()
66     {
67         cout << "~Child() " << ms << endl;
68     }
69 };
70 
71 int main()
72 {       
73     Child cc("cc");
74     
75     cout << endl;
76     
77     return 0;
78 }

      

7,小结:

    1,子类对象在创建时需要调用父类构造函数进行初始化;

    2,先执行父类构造函数然后执行成员的析构函数;

    3,父类构造函数显示调用需要在初始化列表中进行;

    4,子类对象在销毁时需要调用父类析构函数进行清理;

    5,析构顺序与构造顺序对称相反;

       1,先自己,后客人,再父母;

以上是关于C++继承中的构造和析构的主要内容,如果未能解决你的问题,请参考以下文章

C++ 继承中构造和析构对象(面试题可能问)

C++ 类的继承三(继承中的构造与析构)

c++ 复制构造函数和析构函数

小白学习C++ 教程十二C++面向对象编程中的构造函数和析构函数

构造函数和析构函数能不能被继承

c++中的构造函数和析构函数