条款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,可以保持封装性。我们只给外界提供函数调用的形式访问变量,这样的话,无论函数内部做什么改动,对外界都没有影响,外界只取结果即可。试想,如果成员变量直接供外界访问,如果我们不想要这个变量了或者想换个名字,外界调用的地方都需要更改,成本极大。