C++ 虚拟析构函数 (virtual destructor)

举报
我是小白呀iamarookie 发表于 2021/09/10 01:13:42 2021/09/10
【摘要】 C++ 虚拟析构函数 概述问题虚析构函数总结 概述 虚析构函数 (virtual destructor) 可以帮我们实现基类指针删除派生类对象. 问题 当我们从派生类的对象从内存中...

C++ 虚拟析构函数

概述

虚析构函数 (virtual destructor) 可以帮我们实现基类指针删除派生类对象.

在这里插入图片描述

问题

当我们从派生类的对象从内存中撤销时会先调用派生的析构函数, 然后再基类的析构函数, 由此就会产生问题:

  • 如果用 new 运算符建立了派生类对象, 并且由一个基类的指针比那里指向该对象
  • 用 delete 运算符撤销对象时, 系统只执行基类的析构函数. 而不执行派生类的析构函数, 派生类对象析构中要求的工作将被忽略

Base 类:

#ifndef PROJECT6_BASE_H
#define PROJECT6_BASE_H

#include <iostream>
using namespace std;

class Base {
public:
    Base() {
        cout << "执行基类构造函数" << endl;
    };
    ~Base() {
        cout << "执行基类析构函数" << endl;
    };
};

#endif //PROJECT6_BASE_H

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

Derived 类:

#ifndef PROJECT6_DERIVED_H
#define PROJECT6_DERIVED_H

#include <iostream>
#include "Base.h"
using namespace std;

class Derived : public Base {
public:
    Derived() {
        cout << "执行派生类构造函数" << endl;
    };
    ~Derived() {
        cout << "执行派生类析构函数" << endl;
    }
};

#endif //PROJECT6_DERIVED_H

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

main:

#include <iostream>
#include "Derived.h"
using namespace std;

int main() {

    Base *pt =new Derived;
    delete pt;

    return 0;
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

输出结果:

执行基类构造函数
执行派生类构造函数
执行基类析构函数

  
 
  • 1
  • 2
  • 3

虚析构函数

当基类的析构函数为虚函数时, 无论指针指的是同一族中的哪一个类对象, 系统会采用动态关联, 掉啊用相应的析构函数, 对该对象进行清理工作. 即先调用了派生类的析构函数, 再调用了基类的析构函数.

Base 类:

#ifndef PROJECT6_BASE_H
#define PROJECT6_BASE_H

#include <iostream>
using namespace std;

class Base {
public:
    Base() {
        cout << "执行基类构造函数" << endl;
    };
    virtual ~Base() {
        cout << "执行基类析构函数" << endl;
    };
};

#endif //PROJECT6_BASE_H

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

Derived 类:

#ifndef PROJECT6_DERIVED_H
#define PROJECT6_DERIVED_H

#include <iostream>
#include "Base.h"
using namespace std;

class Derived : public Base {
public:
    Derived() {
        cout << "执行派生类构造函数" << endl;
    };
    ~Derived() {
        cout << "执行派生类析构函数" << endl;
    }
};

#endif //PROJECT6_DERIVED_H

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

main:

#include <iostream>
#include "Derived.h"
using namespace std;

int main() {

    Base *pt =new Derived;
    delete pt;

    return 0;
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

输出结果:

执行基类构造函数
执行派生类构造函数
执行派生类析构函数
执行基类析构函数

  
 
  • 1
  • 2
  • 3
  • 4

总结

如果将基类的析构函数声明为虚函数时, 由该基类所派生的所有派生类的析构函数也都自动成为虚函数. 即使派生类的析构函数与其基类的构造函数名字不相同.

最好把基类的析构函数声明为虚函数. 即使基类并不需要析构函数, 我们也可以定义一个函数体为空的虚析构函数, 以保证撤销动态分配空间能正确的处理.

注: 构造函数不能声明为虚函数.

文章来源: iamarookie.blog.csdn.net,作者:我是小白呀,版权归原作者所有,如需转载,请联系作者。

原文链接:iamarookie.blog.csdn.net/article/details/116827263

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。