几个问题
一个类的各数据成员的构造顺序?
按他们在类定义中出现的先后顺序:先定义者先构造。
类的对象成员的构造函数与类自身的构造函数的执行顺序?
先执行对象成员的构造函数,再执行类自身的构造函数。
构造顺序与析构顺序的关系?
二者顺序相反:先构造者,后析构。
构造函数和析构函数用来创建和释放该类的对象,当这个类是派生类时,其对象的创建和释放应与其基类对象及成员对象相联系。
在声明派生类时,一般还应当自己定义派生类的构造函数和析构函数,因为构造函数和析构函数是不能从基类继承的
派生类对象的创建和初始化与基类对象的创建和初始化有关。即构造派生类对象时,要对其基类数据成员、所含对象成员的数据成员以及其他的新增数据成员一起进行初始化。这种初始化工作是由派生类的构造函数来完成的。
派生类成员包括两部分:
(1)从基类继承的成员:由基类构造函数完成
(2)自身定义的成员: 由派生类构造函数完成
在派生类中,构造基类数据成员的可能方式:
方式一,在派生类中直接对基类型数据成员初始化:
class BC { public: BC( ) { x = y = -1; } private: int x, y; }; class DC : public BC { public: DC( ) { x = y = -1;//错误,不能构造基类的私有成员 error C2248: “B::x”: 无法访问 private 成员(在“B”类中声明) } private: string S; };
方式二,显示调用基类构造函数
class BC { public: BC( ) { x = y = -1; } private: int x, y; }; class DC : public BC { public: DC( ) { BC( );//这是构造后才调用,语义错误 } private: string S; };
这样的程序可以编译通过,但语意错误,这是派生类先构造后,在调用基类的构成函数。
正确构成基类数据成员的方式为:
class BC { public: BC( ) { x = y = -1; } private: int x, y; }; class DC : public BC { public: DC( ) : BC( ), S("派生类"), { }//初始化列表 private: string S; };
在创建派生类的对象时,需要调用基类的构造函数:初始化派生类对象从基类继承的成员。在执行一个派生类的构造函数之前,总是先执行基类的构造函数。
调用基类构造函数的两种方式:
(1)显式方式:在派生类的构造函数中,通过参数化表为基类的构造函数提供参数
derived::derived(arg_derived-list):base(arg_base-list)
(2)隐式方式:在派生类/基类的构造函数都缺省时,派生类的构造函数则自动调用基类的默认构造函数。
在一个多层次的继承层次结构中,一个派生类对象的创建时,其构造函数的调用有点类似于多米诺骨牌效应 (domino effect)
列出了不同情况下的派生类构造函数要求:
上面的例子其实也很好理解,我们知道:在执行一个派生类的构造函数之前,总是先执行基类的构造函数。
1.如果基类中无构造函数,那么对于派生类来说,不管派生类是何种构造函数,编译器都会先调用的执行基类的缺省构造函数,然后再执行派生类的构造函数。
2.如果基类是一个无缺省参数的构造函数,那么对于派生类一旦没有构造函数,那么就不会自动的先构造基类的构造函数,这是不允许的。
3.如果基类中有缺省参数的构造函数B(),那么派生类中没有构造函数也是允许的,编译器会自动调用。
通常, 一个基类有一个缺省构造函数。
以下做法是有其实际意义的:当一个派生类对象被创建时会引起某个基类的构造函数的执行。
(这条建议在派生类新增成员依赖于基类成员时体现非常明显)
class Team { public: Team(int len =100) { names = new string[maxno = len ]; //基类构造函数完成其成员初始化,供派生类构造使用。 } protected: string* names; int maxno; }; class BaseballTeam : public Team { public: BaseballTeam(const string s[], int si) : Team(si)//为支持派生类构造本意,必须明确调用基类构造函数。 { for(int i=0; i<si; i++) names[i] = s[i]; //派生类构造函数执行前,基类构造必须完成。 } };
执行构造函数的顺序:
1. 基类的构造函数
2. 子对象的构造函数
3. 其他数据成员初始化
继承下的析构函数 Destructors Under Inheritance
class BC { public: BC( ) { sBC = new char[3]; cout << "BC allocates 3 bytes. "; } ~BC( ) { delete [ ] sBC; cout << "BC free 3 bytes. "; } private: char* sBC; }; class DC : public BC { public: DC( ) { sDC = new char[5]; cout << "DC allocates 5 bytes. "; } ~DC( ) { delete [ ] sDC; cout << "DC free 5 bytes. "; } private: char* sDC; }; int main( ) { DC d; cout << “-------” << endl; return 0; }