条款9:绝不在构造和析构过程中调用virtual函数
不在构造函数和析构函数中调用虚函数,这样的调用不会带来预想的结果。
class Base { public: Base() { cout << "Base default constructor" << endl; cout << "Base address: " << this << endl; func(); } ~Base() { cout << "Base destructor" << endl; } virtual void func(); }; void Base::func() { cout << this << " Base func()" << endl; } class Derived : public Base{ public: Derived() { cout << "Derived defalut constructor" << endl; cout << "Derived address: " << this << endl; } ~Derived() { cout << "Derived destructor" << endl; } void func() override; }; void Derived::func() { cout << this << " Derived func()" << endl; } int main() { Derived derived; return 0; }
来看一下程序输出:
Base default constructor Base address: 0000003DA50FFAE8 0000003DA50FFAE8 Base func() Derived defalut constructor Derived address: 0000003DA50FFAE8 Derived destructor Base destructor
Derived类继承自Base,因此Base类先于Derived类构造。Base类中调用虚函数时,调用的是自身的虚函数。
base class构造期间virtual函数绝不会下降到derived classes阶层
其实在base class构造期间,virtual函数不是virtual函数
为了避免产生迷惑行为(不清楚调用的到底是哪个虚函数),正确做法是:确定你的构造函数和析构函数都没有(在对象被创建和被销毁期间)调用virtual函数。还有一种做法:将Base类中的函数定义为非虚函数,derived class在构造时传递必要信息给Base构造函数。
总结:你无法使用virtual函数从base classes向下调用,在构造函数期间,可以让derived classes将必要的构造信息向上传递至base class构造函数。
条款22:将成员变量声明为private
将成员变量声明为private,可以保持一致性。因为用户唯一能访问对象的形式就是成员函数,所以在调用过程中就不需要纠结加不加“()”的问题了,都是函数,都加小括号即可。
将成员变量声明为private,可以保持封装性。我们只给外界提供函数调用的形式访问变量,这样的话,无论函数内部做什么改动,对外界都没有影响,外界只取结果即可。试想,如果成员变量直接供外界访问,如果我们不想要这个变量了或者想换个名字,外界调用的地方都需要更改,成本极大。