1、绝不在构造与析构过程中调用virtual函数:
在当前执行的构造与析构函数中,内部的虚函数不会下降到派生类中执行派生类重载的虚函数,因为,此时派生类还没有构建。
//.h
Class A{
A(){funa();}
virtual void funa() { cout<<"A";};
};
Class B:public A{
B() {}
virtual void funa() { cout<<"B";};
};
//.cpp
B b; // 此时输出“A”,因为构造A时,虚函数不会传递。
应用场景:构造不同对象时只记录各自的不同信息。解决方法:首先将虚函数改为非虚函数,然后将派生类需要记录信息传递给基类构造函数,如:
//.h
Class A{
A(const string& a){funa(a);}
void funa(const string& a) { cout<<a;};
};
Class B:public A{
B(const string& a,int b): A(a){...}
};
//.cpp
B b; // 此时输出B自己的信息。
小技术点: 如果一个类的多个构造函数都需要执行部分相同工作,为了避免代码重复的一种方法是把共同的初始化代码放入一个初始化函数中,如init
:
Class A{
public:
A(){init();}
A(const A& a){init();}
void init(){...}
};
条款08:别让异常逃离析构函数
析构函数尽量不处理可能抛出异常的代码,而是将此代码写成一个专门的函数,若必须处理则尽量捕获后吞下或者结束程序。
参考资料:Effective C++ 条款09
总结:
1、绝不在构造与析构过程中调用virtual函数,同时他们调用的函数中也禁止调用。
2、多个构造函数执行部分相同工作时,为了避免代码重复可以把共同的初始化代码放入一个初始化函数init
中。
3、别让异常逃离析构函数。