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

相关文章

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

C++ 各种map特点对比分析

《C++各种map特点对比分析》文章比较了C++中不同类型的map(如std::map,std::unordered_map,std::multimap,std::unordered_multima... 目录特点比较C++ 示例代码 ​​​​​​代码解释特点比较1. std::map底层实现:基于红黑

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程