C++中的析构函数

2024-05-31 17:28
文章标签 c++ 函数 析构

本文主要是介绍C++中的析构函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

摘要

动态内存管理

关键点

总结


摘要

析构函数(Destructor)是类的一种特殊成员函数,它在对象生命周期结束时被自动调用,用于执行清理操作。用途包括释放动态分配的内存、关闭文件、释放资源等。析构函数的名称与类名相同,但前面加上波浪号(~),且不接受任何参数,也没有返回值。

当`main`函数结束时,`obj`对象的析构函数会被自动调用:

#include <iostream>class MyClass {
public:MyClass() {std::cout << "Constructor called" << std::endl;}~MyClass() {std::cout << "Destructor called" << std::endl;}
};int main() {MyClass obj;return 0;
}

动态内存管理

析构函数通常用于释放在构造函数中分配的动态内存,以避免内存泄漏。

析构函数中使用`delete[]`来释放构造函数中使用`new`分配的内存:

#include <iostream>class MyClass {
private:int* data;
public:MyClass(int size) {data = new int[size];std::cout << "Constructor called" << std::endl;}~MyClass() {delete[] data;std::cout << "Destructor called" << std::endl;}
};int main() {MyClass obj(10);return 0;
}

关键点

1. 析构函数的虚函数特性:

如果一个类有可能被继承,并且会通过基类指针删除派生类对象,那么基类的析构函数应该声明为虚函数。这确保了删除派生类对象时,会正确调用派生类的析构函数。如果基类的析构函数不是虚函数,那么通过基类指针删除派生类对象时,只会调用基类的析构函数,导致资源泄漏。

#include <iostream>class Base {
public:Base() {std::cout << "Base Constructor called" << std::endl;}virtual ~Base() {std::cout << "Base Destructor called" << std::endl;}
};class Derived : public Base {
public:Derived() {std::cout << "Derived Constructor called" << std::endl;}~Derived() {std::cout << "Derived Destructor called" << std::endl;}
};int main() {Base* obj = new Derived();delete obj; // 调用派生类的析构函数return 0;
}

2. 避免析构函数中抛出异常:

析构函数中如果抛出异常的话,可能会导致程序崩溃。因为在堆栈展开(stack unwinding)过程中,如果析构函数再次抛出异常,会导致`std::terminate`被调用。因此,析构函数中应尽量避免抛出异常,即使可以通过`noexcept(false)`明确声明析构函数可能抛出异常。

#include <iostream>class MyClass {
public:~MyClass() noexcept(false) { // 明确声明可能抛出异常throw std::runtime_error("Exception in Destructor");}
};int main() {try {MyClass obj;} catch (const std::exception& e) {std::cout << "Caught exception: " << e.what() << std::endl;}return 0;
}

3. 对象资源管理:

使用RAIIResource Acquisition Is Initialization)技术,将资源的获取和释放封装在对象的构造和析构函数中,确保资源的正确释放。通过RAII技术,确保在对象生命周期结束时自动关闭文件。

#include <iostream>
#include <fstream>class FileWrapper {
private:std::ofstream file;
public:FileWrapper(const std::string& filename) {file.open(filename);if (!file.is_open()) {throw std::runtime_error("Failed to open file");}}~FileWrapper() {if (file.is_open()) {file.close();}}void write(const std::string& data) {file << data;}
};int main() {try {FileWrapper fw("example.txt");fw.write("Hello, World!");} catch (const std::exception& e) {std::cout << "Caught exception: " << e.what() << std::endl;}return 0;
}

4. 处理资源的共享和转移:

在C++中,使用智能指针(如`std::unique_ptr`和`std::shared_ptr` -- C++中的智能指针类别-CSDN博客)可以自动管理对象的生命周期来自动管理资源,减少手动内存管理的复杂性。

- 使用`std::unique_ptr`管理独占资源。

#include <iostream>
#include <memory>class MyClass {
public:MyClass() {std::cout << "Constructor called" << std::endl;}~MyClass() {std::cout << "Destructor called" << std::endl;}
};void uniquePtrExample() {std::unique_ptr<MyClass> ptr = std::make_unique<MyClass>();
}int main() {uniquePtrExample();return 0;
}

- 使用`std::shared_ptr`管理共享资源。

#include <iostream>
#include <memory>class MyClass {
public:MyClass() {std::cout << "Constructor called" << std::endl;}~MyClass() {std::cout << "Destructor called" << std::endl;}
};void sharedPtrExample() {std::shared_ptr<MyClass> ptr1 = std::make_shared<MyClass>();{std::shared_ptr<MyClass> ptr2 = ptr1;std::cout << "Use count: " << ptr1.use_count() << std::endl;}std::cout << "Use count after scope: " << ptr1.use_count() << std::endl;
}int main() {sharedPtrExample();return 0;
}

总结

1. 析构函数:用于释放资源和执行清理操作,在对象生命周期结束时自动调用。
2. 虚析构函数:在基类中定义虚析构函数,确保通过基类指针删除派生类对象时正确调用派生类的析构函数。
3. 避免异常:析构函数中应避免抛出异常,以防止程序崩溃。
4. RAII:通过RAII技术,将资源管理封装在对象的构造和析构函数中,确保资源的正确释放。
5. 智能指针:使用`std::unique_ptr`和`std::shared_ptr`等智能指针来自动管理对象的生命周期,减少内存管理的复杂性和错误风险。

这篇关于C++中的析构函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1018572

相关文章

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

MySQL高级查询之JOIN、子查询、窗口函数实际案例

《MySQL高级查询之JOIN、子查询、窗口函数实际案例》:本文主要介绍MySQL高级查询之JOIN、子查询、窗口函数实际案例的相关资料,JOIN用于多表关联查询,子查询用于数据筛选和过滤,窗口函... 目录前言1. JOIN(连接查询)1.1 内连接(INNER JOIN)1.2 左连接(LEFT JOI

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

MySQL中FIND_IN_SET函数与INSTR函数用法解析

《MySQL中FIND_IN_SET函数与INSTR函数用法解析》:本文主要介绍MySQL中FIND_IN_SET函数与INSTR函数用法解析,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一... 目录一、功能定义与语法1、FIND_IN_SET函数2、INSTR函数二、本质区别对比三、实际场景案例分