c++复习基础要点03 虚析构函数、虚函数表

1.虚析构函数的作用:

  当基类对象指针通过new动态创建一个子类的对象时,通过该指针释放子类对象时,如果基类的析构函数不是虚函数,则释放该对象时只会调用基类的析构函数而不会调用子类析构函数。这样子类释放对象时就无法释放已分配的资源。

如果基类的析构函数为虚函数,则在基类对象指针去释放子类对象时,就会先调用子类的析构函数,在调用基类的析构函数:

 

#include<iostream>

using namespace std;

 

 

class A

{

public:

       virtual  void fun()

       {

              cout<<"thisis virtual for A"<<endl;

       }

       virtual~A()

       {

              cout<<"thisis virtual ~ for A"<<endl;

       }

};

 

class B:public A

{

public:

       voidfun()

       {

              cout<<"thisis virtual for B"<<endl;

       }

 

       virtual~B()

       {

              cout<<"thisis virtual ~ for B"<<endl;

       }

};

 

 

int main()

{

       A*a;

       a=newB;

       a->fun();

       deletea;

       return0;

}

 

结果:

this is virtual for B

this is virtual ~ for B

this is virtual ~ for A

 

当virtual ~A() 去掉virtual时,结果为:

this is virtual for B

this is virtual ~ for A

 

也就是说,类B的析构函数根本就没被调用,一般情况下类的析构函数里面都是释放内存资源,而析构函数不被调用的话就会造成内存泄漏。

 

所以基类的析构函数应该被定义为虚函数,这样做是为了当用一个基类的指针删除一个派生类的对象时,派生类的析构函数会被调用。

 

 当然,并不是要把所有类的析构函数都写成虚函数。因为当类里面有虚函数的时候,编译器会给类添加一个虚函数表,里面来存放虚函数指针,这样就会增加类的存储空间。所以,只有当一个类被用来作为基类的时候,才把析构函数写成虚函数。

 

2.虚函数表

 

多态性可分为两类:静态多态和动态多态。函数重载和运算符重载实现的多态属于静态多态,动态多态性是通过虚函数实现的。

每个含有虚函数的类有一张虚函数表(vtbl),表中每一项是一个虚函数的地址, 也就是说,虚函数表的每一项是一个虚函数的指针。

没有虚函数的C++类,是不会有虚函数表的

 

例子:

class Class1

{

 public:

    m_data1;

    m_data2;

    virtual  vfunc1();

    virtual  vfunc2();

    virtual  vfunc3();

};

 

图示:

c++复习基础要点03 虚析构函数、虚函数表_第1张图片

 

class Class2 : public Class1

{

 public:

    m_data3;

    virtual vfunc2();

 

};

 

图示:

c++复习基础要点03 虚析构函数、虚函数表_第2张图片

虚函数表的指针4个字节大小(vptr),存在于对象实例中最前面的位置(这是为了保证取到虚函数表的有最高的性能——如果有多层继承或是多重继承的情况下)。这意味着我们通过对象实例的地址得到这张虚函数表,然后就可以遍历其中函数指针,并调用相应的函数。

 

 

你可能感兴趣的:(多态,虚函数表,虚析构)