浅谈C++--绝不在构造和析构函数中调用virtual函数

C++规定了虚函数的行为,但将实现方法留给了编译器作者。

编译器处理虚函数的方法是:给每个对象添加一个隐藏成员。隐藏成员中保存了一个指向函数地址数组的指针,这种数组成为虚函数表。

这篇博客我们不是试图详细说明虚函数的工作原理。

但是我们也应该知道,使用虚函数,在内存和执行速度方面有一定的成本:
每个对象都将增大、增大量为存储地址的空间;
对于每个类,编译器都创建一个虚函数地址表;
对于每个函数调用,都需要执行一项额外的操作,即到列表中查找地址。

下面看一下虚函数的其他知识:
1.构造函数不能是虚函数
2.如果做基类,析构函数应当是虚函数
3.友元不能是虚函数
因此友元不是类成员,而只有类成员函数才能是虚函数。
4.没有重新定义,不应该声明为虚函数

现在我们要出大招了!
我们定义一个类:

class Transaction
{
public:
    Transaction();
    virtual void logTransaction() const = 0;
    ...
};
Transaction::Transaction()
{
    ...
    Transaction();
}
class BuyTransaction:public Transaction
{
public:
    virtual void logTransaction()const;
    ...
};
class SellTransaction:public Transaction
{
public:
    virtual void logTransaction() const;
};

暴行来了:

BuyTransaction b;

无疑在BuyTransaction构造函数调用之前会调用基类Transaction的构造函数。
Transaction构造函数最后一行调用虚函数logTransaction。
理智分析:基类构造期间虚函数绝不会下降到子类阶层。就好比对象内部使用了尚未初始化的成分。虚函数会被编译器解析到基类。

相同的道理同样适用于析构函数。

所以切忌!
确定你的构造函数和析构函数都没有调用virtual函数,而他们调用的所有函数也都服从同一约束!

你可能感兴趣的:(C++,virtual)