条款9和条款22

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

上一篇:Android


下一篇:JAVA