析构函数定位虚函数的作用

析构函数定位虚函数的作用
当一个继承类经由一个基类的指针删除时,而该基类包含的是一个非虚析构函数,其结果是未定义的(实际执行时通常发生的是继承类的独有成分没有被销毁)。这个后果很严重,会造成**内存泄漏**。

通过基类指针删继承类时,需要特别注意!!

/*
* 当一个继承类经由一个基类的指针删除时,而该基类包含的是一个非虚析构函数,
* 其结果是未定义的(实际执行时通常发生的是继承类的独有成分没有被销毁)。
* 这个后果很严重,会造成内存泄漏。
*
* 通过基类指针删继承类时,需要特别注意!!
*/

#include <iostream>
using namespace std;

class Base{
public:
    Base (){
        cout<<"Base 构造函数执行了!"<<endl;
    }
    virtual ~Base(){
        cout<<"Base 析构函数执行了!"<<endl;
    }
    virtual void func(){
        cout<<"Base::func 虚函数执行了!"<<endl;
    }
};
class Derived :public Base
{
public:
    Derived(){
        cout<<"Derived 构造函数执行了!"<<endl;
    }
    ~Derived(){
        cout<<"Derived 析构函数执行了!"<<endl;
    }
    virtual void func(){
        cout<<"Derived::func 虚函数执行了!"<<endl;
    }
};

int main()
{
    Base *p;
    p=new Derived;
    p->func();

    delete  p;      //!此时只会执行Base的析构函数,不会执行Dervied的析构函数,从而导致内存泄露;
                        //!解决方法==>>将基类的析构函数定义为虚函数;
    return 0;
}

//------------------------------------输出信息-----------------------------------//
Base 构造函数执行了!               //先构造基类
Derived 构造函数执行了!            //再构造继承类
Derived::func 虚函数执行了!
Derived 析构函数执行了!            //先析构继承类
Base 析构函数执行了!               //在析构子类

//------------------------------------主函数修改-----------------------------------//
int main()
{
    Derived d;
    d.func();
}
//------------------------------------输出信息-----------------------------------//
Base 构造函数执行了!
Derived 构造函数执行了!
Derived::func 虚函数执行了!
Derived 析构函数执行了!
Base 析构函数执行了!

//==>> 局部对象在声明结束时自动析构;
//==>> 在构造对象的时候,先构造基类,再构造继承类;在析构的时候,先析构继承类,再析构基类;即先构造的后析构;

//------------------------------------主函数修改-----------------------------------//
int main()
{
    Derived *d2=new Derived;
    d2->func();
}
//------------------------------------输出信息-----------------------------------//
Base 构造函数执行了!
Derived 构造函数执行了!
Derived::func 虚函数执行了!

//==>> 不会执行析构函数!!
//==>> 所以,通过new在堆上划分的内存必须需要通过手动释放,不然会导致内存泄露!!==>delete d2;
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中的虚函数是通过在基类中使用关键字virtual来声明的函数。虚函数可以在派生类中被重写,使得在运行时根据对象的类型动态调用正确的函数。虚函数的调用是基于对象的动态类型实现的,这意味着即使使用基类的指针或引用来调用虚函数,实际调用的仍然是对象的派生类中的函数。 纯虚函数是在基类中声明但没有实现的虚函数。纯虚函数的声明以 "= 0" 结尾,表示该函数没有函数体。纯虚函数的存在使得基类成为抽象类,抽象类不能被实例化。派生类必须实现基类中的纯虚函数才能被实例化。 构造函数不能被声明为虚函数,因为虚函数的调用依赖于对象的类型,而在构造函数中对象的类型尚未确定。因此,构造函数无法使用虚函数的动态派发机制。 析构函数可以被声明为虚函数,当基类指针指向派生类对象并通过该指针删除对象时,如果析构函数不是虚函数,那么只会调用基类的析构函数而不会调用派生类的析构函数,从而导致派生类对象没有正确地被销毁。使用虚析构函数可以确保在删除对象时正确定位到派生类的析构函数,并按照正确的顺序销毁对象。 构造函数和析构函数都可以调用虚函数,但需要注意的是,在构造函数中调用虚函数时可能会导致意外的行为,因为在构造函数执行期间,对象的派生类部分尚未初始化。因此,最好在构造函数中避免调用虚函数。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C++虚函数和纯虚函数的问题总结](https://blog.csdn.net/weixin_44477424/article/details/124526204)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值