一、不要将构造函数定义为虚函数
1、从存储空间角度
每一个虚函数都对应一个vtable(虚函数表),这个vtable就是存储在对象的内存空间的,那么如果构造函数是虚的,就需要通过vtable来调用,可是对象还没有实例化,即内存空间还没有,无法找到vtable,所以构造函数不能是虚的。
2、从对象类型角度
构造一个对象的时候,必须知道对象的实际类型,而虚函数行为是在运行期间确定实际类型的,是该类本身,还是该类的一个派生类,或是更深层次的派生类,无法确定。
3、从实现角度
虚函数的执行依赖于这个vtable。同时vtable在构造函数调用后才建立,因而构造函数不可能成为虚函数。
二、最好将基类的析构函数定义为虚函数
因为在类的继承中,如果有基类指针或引用指向派生类,那么当我们用基类指针delete时,如果不定义成虚函数,派生类中派生的那部分无法析构,因此会造成内存泄漏。
三、要在构造函数和析构函数中调用虚函数
在基类构造的时候,虚函数是非虚的,不会走到派生类中,既采用的静态绑定。显然,当我们构造一个子类对象时,先调用基类的构造函数,构造子类中基类部分,子类还没有构造,没有初始化,如果在基类的构造函数中调用虚函数,如果可能的话会调用一个还没有初始化的对象,那是很危险的,所以在c++中不可以在构造父类对象部分的时候调用子类的虚函数实现。但是如果你这么做了编译器也不会报错,只是话编译器不会给你调用子类的实现,而是依旧调用基类的实现。
析构的时候首先调用子类的析构函数,析构掉对象中的子类部分,然后再调用基类的析构函数析构基类部分。如果在基类的析构函数中调用虚函数,会导致其调用已经析构了的子类对象里面的函数,这是非常危险的。