[Effective C++]条款09:绝不在构造和析构过程中调用virtual函数

1、在C++中,应该避免在类的构造函数和析构函数中调用虚函数
原因:

  • 创建对象时,先执行基类的构造函数,后执行派生类构造函数,基类的构造函数执行时,对象的派生类部分还未创建。基类构造函数执行时,派生类的成员变量和虚函数表(vtable)尚未初始化。虚函数最终执行的是基类版本,而不是派生类的版本
  • 析构函数中一样的道理,析构函数执行顺序,是先执行派生类析构函数,后执行基类析构函数,当基类析构函数执行时,派生类对象已经销毁了。
    例子:
#include <iostream>

class Base {
public:
    Base() {
        // 在构造函数中调用虚函数
        print();
    }

    virtual ~Base() {
        // 在析构函数中调用虚函数
        print();
    }

    virtual void print() const {
        std::cout << "Base::print()" << std::endl;
    }
};

class Derived : public Base {
public:
    Derived() {
        print();
    }

    ~Derived() {
        print();
    }

    void print() const override {
        std::cout << "Derived::print()" << std::endl;
    }
};

int main() {
    Derived d;
    return 0;
}
输出:
- 预想结果是派生类实现了虚函数print,应该打印的都是Derived::print(),但构造函数调用的是Base的print函数
Base::print()
Derived::print()
Derived::print()
Base::print()

2、解决方法
2.1、使用非虚函数:

  • 如果需要在构造函数或析构函数中调用某个函数,可以将其定义为非虚函数,或者使用模板方法模式。
  • 如果在构造函数中调用了派生子类复写后的非虚函数,那最终调用的是父类指向的方法,因为子类对象还未创建

2.2、在构造函数完成后调用虚函数:

  • 将虚函数的调用一道构造函数之外,确保对象已经完全构造。这才是正解

2.3、使用工厂函数:

  • 通过工厂函数创建对象,并在对象完全构造后调用虚函数,和上面方式一个道理。先确保对象创建,再调用虚函数。
#include <iostream>

class Base {
public:
    Base() {
        // 避免在构造函数中调用虚函数
    }

    virtual ~Base() {
        // 避免在析构函数中调用虚函数
    }

    void initialize() {
        // 在对象完全构造后调用虚函数
        print();
    }

    virtual void print() const {
        std::cout << "Base::print()" << std::endl;
    }
};

class Derived : public Base {
public:
    Derived() {
    }

    ~Derived() {
    }

    void print() const override {
        std::cout << "Derived::print()" << std::endl;
    }
};

int main() {
    Derived d;
    d.initialize();  // 在对象完全构造后调用虚函数
    return 0;
}
输出:Derived::print()

3、总结:

  • 在构造和析构函数期间不要调用virtual函数,因为基类构造函数方法调用时,子类还未创建。

思维导图笔记
请添加图片描述

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值