C++虚函数和虚析构函数小结

虚函数是这么定义的:

被virtual关键字修饰的成员函数,虚函数用来实现运行时多态,指向基类的指针在操作它的多态类对象时,会根据不同的类对象,调用其相应的函数,这个函数就是虚函数。

我们先来看一个例子吧

#include<iostream>
class person{
public:
    person(){};
    void say_something(){
        std::cout<<"I am a handsome man!"<<std::endl;
    }
};
class zhouchaoyuan:public person{
public:
    zhouchaoyuan(){};
    void say_something(){
        std::cout<<"I am zhouchaoyuan"<<std::endl;
    }
};
int main(){
    person object1;
    zhouchaoyuan object2;
    person *p1=&object1;
    person *p2=&object2;
    p1->say_something();
    p2->say_something();
    return 0;
}

结果居然是:

I am a handsome man!
I am a handsome man!

这个就不是我们所期望的,这个时候虚函数就派上用场了,我们在

public:
    person(){};
    virtual void say_something(){
        std::cout<<"I am a handsome man!"<<std::endl;
    }
};

也就是说吧say_something声明成虚函数,这个时候输出的结果就是:

I am a handsome man!
I am zhouchaoyuan

有没有感觉好神奇,为什么会这样呢?

事实上我们我们声明了虚函数之后没有对象都会创建一个虚表,我们称之为vtbl,而且每个对象会有一个虚表指针我们称之为vptr,分别指向这个虚表,虚表里面存有虚函数的入口地址,不同的对象有不同的vptr,只要vptr不同,指向的vtbl就不同,而不同的vtbl里装着对应类的虚函数地址,这样就可以找到我们要的虚函数完成它的任务。


接下来我们看看虚析构函数,重写上述代码如下

#include<iostream>
class person{
public:
    person(){};
    ~person(){
        std::cout<<"The destructor in person"<<std::endl;
    };
    virtual void say_something(){
        std::cout<<"I am a handsome man!"<<std::endl;
    }
};
class zhouchaoyuan:public person{
public:
    zhouchaoyuan(){};
    ~zhouchaoyuan(){
        std::cout<<"The destructor in zhouchaoyuan"<<std::endl;
    };
    void say_something(){
        std::cout<<"I am zhouchaoyuan"<<std::endl;
    }
};
int main(){
    person *p=new zhouchaoyuan();
    p->say_something();
    delete p;
    return 0;
}
惊奇的发现输出:

I am zhouchaoyuan
The destructor in person

当用一个基类的指针删除一个派生类的对象时,派生类的析构函数不会被调用。而我们的析构函数是用来回收内存的,如果析构函数没有被调用,那就是说当前对象占用的内存不会被回收,也就是说造成了内存泄露,这个是很危险的。

这个时候我们可以很容易的解决这个问题,那就是吧基类的析构函数定义成虚函数,声明如下:

class person{
public:
    person(){};
    virtual ~person(){
        std::cout<<"The destructor in person"<<std::endl;
    };
    virtual void say_something(){
        std::cout<<"I am a handsome man!"<<std::endl;
    }
};

I am zhouchaoyuan
The destructor in zhouchaoyuan
The destructor in person

好了,好像看到了比较满意的结构了,因为子类终于被回收了!!!!总的来说虚析构函数就是为了当用一个基类的指针删除一个派生类的对象时,派生类的析构函数会被调用。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值