虚析构函数

在 C++ 中,将析构函数声明为 virtual 是一种常见的做法,特别是在使用继承的情况下。这么做的主要原因是确保当你删除一个指向派生类对象的基类指针时,能够正确调用派生类的析构函数,从而防止资源泄漏。下面详细解释为什么要在基类中将析构函数声明为 virtual。

基类与派生类

假设有一个基类 CBaseHmi 和一个派生类 CHmi:

class CBaseHmi {
public:
    virtual ~CBaseHmi() = default; // 虚析构函数
};

class CHmi : public CBaseHmi {
public:
    CHmi();
    virtual ~CHmi() = default;//显示 virtual,只要基类加了,这里加不加都是正确调用派生类的析构函数
};

为什么析构函数要声明为 virtual

当一个类有虚函数时(包括虚析构函数),它会在对象中维护一个虚函数表(vtable),这个表包含了类的虚函数指针。当你通过基类指针删除一个对象时,虚函数机制确保调用的是最派生类的析构函数。

示例解释

void example() {
    CBaseHmi* obj = new CHmi();
    delete obj; // 正确调用 CHmi 的析构函数
}

在这个示例中:

  • obj 是一个指向 CBaseHmi 类型的指针,但它实际上指向一个 CHmi 对象。
  • 如果 CBaseHmi 的析构函数不是虚函数,那么 delete obj 只会调用 CBaseHmi 的析构函数,而不会调用 CHmi
    的析构函数。这可能会导致资源泄漏,因为 CHmi 的析构函数中可能有清理特定资源的代码。

没有虚析构函数的情况下

如果没有虚析构函数,派生类的析构函数不会被调用。例如:

class CBaseHmi {
public:
    ~CBaseHmi() = default; // 非虚析构函数
};

class CHmi : public CBaseHmi {
public:
    CHmi();
    ~CHmi() { // 派生类析构函数
        std::cout << "CHmi destructor" << std::endl;
    }
};

void example() {
    CBaseHmi* obj = new CHmi();
    delete obj; // 只调用 CBaseHmi 的析构函数
}

在这个例子中,输出不会包含 “CHmi destructor”,因为 CHmi 的析构函数没有被调用。

使用虚析构函数

通过将基类的析构函数声明为虚函数,可以确保正确调用派生类的析构函数:

class CBaseHmi {
public:
    virtual ~CBaseHmi() = default; // 虚析构函数
};

class CHmi : public CBaseHmi {
public:
    CHmi();
    ~CHmi() { // 派生类析构函数
        std::cout << "CHmi destructor" << std::endl;
    }
};

void example() {
    CBaseHmi* obj = new CHmi();
    delete obj; // 正确调用 CHmi 的析构函数
}

在这个例子中,输出将包含 “CHmi destructor”,因为 CHmi 的析构函数被正确调用。

总结

将析构函数声明为 virtual 以确保:

  • 当通过基类指针删除对象时,能够正确调用派生类的析构函数。
  • 防止资源泄漏和未定义行为。

这是面向对象编程中管理资源和多态性的关键机制之一。

  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

王成长

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值