ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

多态中的虚析构函数

2019-08-30 13:04:58  阅读:217  来源: 互联网

标签:析构 函数 多态 virtual 调用 Animal COM 虚析构


为什么析构函数要声明成virtual呢?

因为,如果delete一个基类的指针时, 如果它指向的是一个子类的对象,那么析构函数不为虚就会导致无法调用子类析构函数,从而导致资源泄露。

如果一个类要被使用成多态的,那么这个virtual是必须的。比如:

#include <iostream>
using namespace std;

class Animal
{
    char* ap;
public:
    Animal()
    {
        ap = new char;
        std::cout << "Animal ctor" << std::endl;
    }
    virtual void foo()
    {
        std::cout << "Animal::foo" << std::endl;
    }
    virtual ~Animal()
    {
        std::cout << "Animal dtor" << std::endl;
        delete ap;
    }
};
class Dog : public Animal
{
    char* dp;
public:
    Dog()
    {
        dp = new char;
        std::cout << "Dog ctor" << std::endl;
    }
    virtual void foo()
    {
        std::cout << "Dog::foo" << std::endl;
    }
    virtual ~Dog()
    {
        delete dp;
        std::cout << "Dog dtor" << std::endl;
    }
};

int main()
{
    Animal* pa = new Dog();
    pa->foo();
    delete pa;
    return 0;
}

delete pa 实际上相当于:

pa->~Animal();

释放pa所指向的内存

在这里,因为~Animal()是virtual的,尽管是通过Animal类型的指针调用的,根据虚表v-table的信息,~Dog()被正确调用到。如果把virtual属性去掉,那么被调用的是~Animal(),Dog类的构造函数被调用而析构函数未被调用,构造函数中分配的资源没有释放,从而产生了内存泄漏。析构函数缺省声明为virtual,就可以避免这一问题。

如果基类析构函数不加virtual,运行效果如下:

Animal ctor
Dog ctor
Dog::foo
Animal dtor

可另一个问题是,有时virtual是不需要的。

如果一个类不会被继承,比如一个utility类,该类完全是静态方法;

或者一些类尽管可能会被继承,但不会被使用成多态的,即除了析构函数外,没有其他的方法是virtual的,这时就可以把virtual属性去掉。

去掉析构函数的virtual属性后,因为该类中没有其他的virtual函数,所以编译时不会生成v-table,这样就节省了编译时间,并减少了最终生成的程序的大小。更重要的是,遵从这一规则,给该类的维护者一个信息,即该类不应被当作多态类使用。

同样,当作一个抽象时,如果你模仿Java的interface,声明了如下的虚基类:

class AbstractBase
{
    virtual method1() = 0;
    virtual method2() = 0;
};

那么应该给它增加一个空的virtual析构函数:

virtual ~AbstractBase(){}

如果你对COM比较熟悉,可能会注意到,COM interface中并没有这个virutal构造函数。这是因为,COM通过使用引用计数的机制来维护对象。当你使用完一个COM对象,调用Release()时,COM的内部实现检查引用技术是否为零。如果是,则调用:

delete this;

因为Release()是virtual的,所以该COM对象对应的正确的派生类被调用,delete this会调用正确的析构函数,达到了使用virtual析构函数的效果。

标签:析构,函数,多态,virtual,调用,Animal,COM,虚析构
来源: https://www.cnblogs.com/WindSun/p/11434408.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有