【C++入门到精通】C++入门 —— 类和对象(构造函数、析构函数)下

简介: 【C++入门到精通】C++入门 —— 类和对象(构造函数、析构函数)

三、析构函数


    ⭕析构函数概念


       析构函数:与构造函数功能相反,析构函数不是完成对对象本身的销毁,局部对象销毁工作是由编译器完成的。而对象在销毁时会自动调用析构函数,完成对象中资源的清理工作。它的主要作用是确保对象使用的资源被正确释放,防止资源泄漏和内存泄漏的发生。


   ⭕析构函数的特点


1. 析构函数的名称与类名相同,但前面有一个波浪线(~)作为前缀。例如,如果类名为MyClass,则对应的析构函数名为~MyClass。

2. 析构函数没有返回类型,包括void,因为它不返回任何值。

3. 析构函数不接受任何参数,因此不能被重载,每个类只能有一个析构函数。

4. 析构函数不能手动调用,它会在对象生命周期结束时自动被触发执行。对象销毁的情况包括对象超出作用域、通过delete释放了对应的动态分配内存、容器中的对象被移除等。

5. 当对象的析构函数被调用时,它会自动调用成员对象的析构函数,并按照它们被定义的顺序进行销毁。

6. 如果没有显式定义析构函数,编译器会默认生成一个空的析构函数,即不做任何操作。


       析构函数通常用于在对象销毁时执行一些必要的清理工作,比如释放动态分配的内存、关闭文件句柄、释放其他类型的资源等。通过析构函数,可以确保对象所使用的资源被正确释放,避免资源泄漏和内存泄漏的问题。


下面是一个简单的示例,展示了一个类FileHandler的析构函数的定义和使用


#include <iostream>
#include <fstream>
class FileHandler {
public:
    FileHandler(const std::string& filename) {
        file.open(filename);
        if (!file) {
            std::cout << "Failed to open file " << filename << std::endl;
        }
    }
    ~FileHandler() {
        if (file.is_open()) {
            file.close();
            std::cout << "File closed." << std::endl;
        }
    }
    void writeToFile(const std::string& data) {
        file << data << std::endl;
    }
private:
    std::ofstream file;
};
int main() {
    FileHandler handler("example.txt");
    handler.writeToFile("Hello, world!");
    return 0;
}

       在上面的示例中,FileHandler类封装了一个文件句柄,其中构造函数用于打开文件并检查是否成功打开,而析构函数则负责在对象销毁时关闭文件。在main函数中,创建了一个FileHandler对象handler并写入数据,当对象handler超出作用域时,析构函数会被自动调用,关闭文件。


       通过定义适当的析构函数,可以确保对象的资源在对象销毁时被正确释放,从而保证程序运行的安全性和可靠性。也值得注意的是,在使用动态分配的内存或其他相关资源时,需要显式定义析构函数,以免造成资源泄漏。


⭕常见析构函数的几种类型


根据特定的需求和情况,常见的析构函数可以分为以下四种类型:


       1. 默认析构函数(Default Destructor):如果没有显式定义析构函数,编译器会隐式地生成一个默认的析构函数。默认析构函数没有任何实现内容,它的作用仅仅是调用成员对象的析构函数,并按照它们的定义顺序进行销毁。


class MyClass {
    // 在这里没有显式定义析构函数
    // 编译器会自动生成一个默认析构函数
};

       2. 虚析构函数(Virtual Destructor):当一个类被继承时,通常应该考虑使用虚析构函数。虚析构函数在基类中声明为虚函数,可以确保当通过指向基类的指针删除一个基类对象时,实际上会调用到派生类的析构函数。这是为了确保析构函数正确释放所有的资源,包括派生类自己的资源。


class MyBaseClass {
public:
    virtual ~MyBaseClass() { // 声明为虚函数
        // 析构函数的实现
    }
};
class MyDerivedClass : public MyBaseClass {
public:
    ~MyDerivedClass() {
        // 派生类析构函数的实现
    }
};

       3. 纯虚析构函数(Pure Virtual Destructor):纯虚析构函数是一个没有具体实现的虚析构函数。它的存在是为了将基类定义为抽象类,即不能实例化的类。纯虚析构函数要求派生类必须提供具体的实现。


class AbstractClass {
public:
    virtual ~AbstractClass() = 0; // 纯虚析构函数,没有实现
};
AbstractClass::~AbstractClass() {
    // 纯虚析构函数的实现
}
class ConcreteClass : public AbstractClass {
public:
    ~ConcreteClass() {
        // 派生类的析构函数实现
    }
};

       4. 自定义析构函数(Custom Destructor):自定义析构函数根据类的具体需求定义。它可以用于释放动态分配的内存、关闭文件句柄、释放其他资源等。自定义的析构函数应该根据类的设计在适当的时候进行资源的清理和回收。


class MyClass {
public:
    ~MyClass() {
        // 自定义的析构函数实现
        // 释放资源,关闭文件等
    }
};

       要根据你的需求选择适当的析构函数类型。大多数情况下,默认析构函数已足够满足基本需求,但当涉及到多态性和继承时,可能需要使用虚析构函数。对于抽象类,可以使用纯虚析构函数。自定义析构函数提供了更灵活的资源清理能力。


四、温馨提示


       感谢您对博主文章的关注与支持!在阅读本篇文章的同时,非常感谢您留下您宝贵的意见和反馈。如果您喜欢这篇文章,可以点赞、评论和分享给您的同学,这将对我提供巨大的鼓励和支持。另外,博主计划在未来的更新中持续探讨与本文相关的内容。我会为您带来更多关于C++以及编程技术问题的深入解析、应用案例和趣味玩法等。请继续关注博主的更新,不要错过任何精彩内容!


       我将会不断为大家带来更多精彩、有趣的文章和内容。与这篇文章相关的内容也将陆续推出,希望你们能够一直关注我们的动态。余下的拷贝构造函数、赋值运算符重载、const成员、取地址及const取地址操作符重载详见下回分解下一篇链接:拷贝构造函数、赋值运算符重载、const成员函数


       再次感谢您的支持和关注。我们期待与您建立更紧密的互动,共同探索C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!

18168aaf881f45ddaf629859c85d9bc7.jpg



目录
相关文章
|
5天前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
15 4
|
5天前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
15 4
|
5天前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
10 1
|
6天前
|
存储 编译器 C语言
【C++打怪之路Lv3】-- 类和对象(上)
【C++打怪之路Lv3】-- 类和对象(上)
13 0
|
15天前
|
存储 编译器 C++
【C++类和对象(下)】——我与C++的不解之缘(五)
【C++类和对象(下)】——我与C++的不解之缘(五)
|
15天前
|
编译器 C++
【C++类和对象(中)】—— 我与C++的不解之缘(四)
【C++类和对象(中)】—— 我与C++的不解之缘(四)
|
17天前
|
编译器 C语言 C++
C++入门3——类与对象2-2(类的6个默认成员函数)
C++入门3——类与对象2-2(类的6个默认成员函数)
22 3
|
17天前
|
存储 编译器 C语言
C++入门2——类与对象1(类的定义和this指针)
C++入门2——类与对象1(类的定义和this指针)
20 2
|
17天前
|
C++
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
49 1
|
17天前
|
编译器 C语言 C++
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
15 1