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

相关文章

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决