C++ RAII

RAII定义

RAII(Resource Acquisition Is Initialization)是C++编程中的一种重要的资源管理技术。它的核心思想是:资源的获取应该在对象的构造阶段进行,而资源的释放则应该在对象的析构阶段进行。通过利用C++对象的生命周期和析构函数,在对象生命周期结束时自动释放资源,从而避免资源泄漏和内存泄漏的发生。
具体来说,RAII 的实现方式是将资源的管理封装到类中,利用类的构造函数来获取资源,利用析构函数来释放资源。这样,当对象被创建时,资源被获取;当对象被销毁时,资源会自动释放,即使因为异常或者其他原因导致函数提前返回,也能够保证资源被正确释放,从而确保资源的正确管理。


示例

文件打开

下面是一个简单的示例,演示了如何使用 RAII 来管理文件资源:

#include <iostream>
#include <fstream>
#include <string>
#include <stdexcept>class FileResource {
private:std::ofstream file; // 文件资源std::string filename; // 文件名public:FileResource(const std::string& filename) : filename(filename) {file.open(filename); // 在构造函数中打开文件if (!file.is_open()) {throw std::runtime_error("Failed to open file: " + filename);}std::cout << "File " << filename << " opened successfully." << std::endl;}~FileResource() {if (file.is_open()) {file.close(); // 在析构函数中关闭文件std::cout << "File " << filename << " closed." << std::endl;}}// 写入数据到文件void writeData(const std::string& data) {if (!file.is_open()) {throw std::runtime_error("File is not open.");}file << data;}
};int main() {try {FileResource file("example.txt"); // RAII:文件资源在构造函数中获取,在析构函数中释放// 在文件中写入数据file.writeData("Hello, RAII!");} catch (const std::exception& e) {std::cerr << "Exception: " << e.what() << std::endl;}return 0;
}

在这个示例中,FileResource类封装了文件资源,它的构造函数负责打开文件,而析构函数负责关闭文件。当FileResource对象在main函数中创建时,文件被打开,当对象生命周期结束时,文件会自动关闭,即使在函数中抛出异常,文件也能够得到正确的关闭。
这个示例展示了 RAII 的核心思想:利用对象生命周期和析构函数来确保资源的正确获取和释放,从而提高代码的健壮性和可维护性。

锁和内存分配的使用

示例

#include <iostream>
#include <memory>
#include <mutex>// RAII for dynamic memory allocation
class DynamicMemoryResource {
private:int* data;public:DynamicMemoryResource(int size) : data(new int[size]) {std::cout << "Dynamic memory allocated." << std::endl;}~DynamicMemoryResource() {delete[] data;std::cout << "Dynamic memory deallocated." << std::endl;}// Other methods to interact with the allocated memory// ...int getValue(int index) const {return data[index];}void setValue(int index, int value) {data[index] = value;}
};// RAII for locking
class LockResource {
private:std::mutex& mtx;public:LockResource(std::mutex& mutex) : mtx(mutex) {mtx.lock();std::cout << "Mutex locked." << std::endl;}~LockResource() {mtx.unlock();std::cout << "Mutex unlocked." << std::endl;}// Other methods to perform operations while holding the lock// ...
};int main() {try {// RAII for dynamic memory allocationDynamicMemoryResource dynamicMemory(10);// RAII for lockingstd::mutex myMutex;{LockResource lock(myMutex);  // RAII for locking// Perform operations while holding the lockdynamicMemory.setValue(0, 42);std::cout << "Value at index 0: " << dynamicMemory.getValue(0) << std::endl;}// The lock is automatically released when the LockResource object goes out of scope// Other operations after releasing the lock// ...} catch (const std::exception& e) {std::cerr << "Exception: " << e.what() << std::endl;}return 0;
}

RAII 在c++ 标准库中的应用

在C++标准库中,RAII(资源获取即初始化)的理念广泛应用于各种类和功能。以下是一些C++标准库中常见的RAII应用:

智能指针

std::unique_ptr 和 std::shared_ptr 提供了自动管理动态分配的内存资源的机制。当指针超出作用域时,它们会自动释放所持有的内存。

#include <memory>
#include <iostream>int main() {std::unique_ptr<int> ptr(new int(42));std::cout << *ptr << std::endl; // 输出: 42// 在ptr超出作用域后,自动释放所持有的内存
}


文件流

std::ifstream 和 std::ofstream 等文件流类利用RAII来确保在文件操作完成后自动关闭文件。文件资源在对象生命周期结束时被释放。

#include <fstream>int main() {std::ofstream file("example.txt");file << "Hello, RAII!";// file对象超出作用域后,文件自动关闭
}


标准容器

标准容器如 std::vector、std::string 在内部使用RAII原则来管理其元素的内存。当容器对象销毁时,相关的资源被自动释放。

#include <vector>int main() {std::vector<int> vec{1, 2, 3, 4, 5};// vec对象超出作用域后,自动释放内存
}


互斥锁

std::mutex 和 std::lock_guard 用于实现线程同步,其中 std::lock_guard 利用RAII确保在作用域结束时释放锁资源,避免忘记手动释放锁。

#include <mutex>
#include <thread>
#include <iostream>std::mutex mtx;void task() {std::lock_guard<std::mutex> lock(mtx);std::cout << "Critical section" << std::endl;// lock对象超出作用域后,自动释放锁资源
}int main() {std::thread t1(task);std::thread t2(task);t1.join();t2.join();
}


文件系统库

C++17 引入的 <filesystem> 库中的路径、文件迭代器等对象也遵循RAII原则,确保在作用域结束时资源被正确释放。

#include <filesystem>
#include <iostream>namespace fs = std::filesystem;int main() {fs::path filePath = "example.txt";// filePath对象超出作用域后,自动释放资源
}


计时器

std::chrono 库中的定时器类,如 std::chrono::steady_clock::time_point,在其生命周期结束时会自动释放相关资源。

#include <chrono>
#include <iostream>int main() {auto start = std::chrono::steady_clock::now();// 在start超出作用域后,自动释放资源
}

异常安全性

C++标准库中的很多异常安全性保障都使用了RAII,例如 std::lock_guard 在异常发生时仍能正确释放锁资源,确保不会发生资源泄漏。

#include <iostream>
#include <stdexcept>int main() {try {// 执行一些可能抛出异常的操作throw std::runtime_error("An error occurred");} catch(const std::exception& e) {std::cerr << "Exception caught: " << e.what() << std::endl;// 在catch块中,资源会被正确释放}
}


线程

std::thread 类在其析构函数中处理了线程资源的清理,确保在线程对象销毁时相关资源被释放。

#include <thread>
#include <iostream>void task() {std::cout << "Thread task" << std::endl;
}int main() {std::thread t(task);t.join();// t对象超出作用域后,线程资源会被正确释放
}

这些都是C++标准库中使用RAII的一些常见例子。通过RAII,C++标准库实现了自动化的资源管理,提高了代码的可维护性和安全性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/696945.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C#之WPF学习之路(2)

目录 控件的父类 DispatcherObject类 DependencyObject类 DependencyObject 类的关键成员和方法 Visual类 Visual 类的主要成员和方法 UIElement类 UIElement 类的主要成员和功能 FrameworkElement类 FrameworkElement 类的主要成员和功能 控件的父类 在 WPF (Windo…

谷粒商城篇章9 ---- P248-P261/P292-P294 ---- 消息队列【分布式高级篇六】

目录 1 消息队列(Message Queue)简介 1.1 概述 1.2 消息服务中两个重要概念 1.3 消息队列主要有两种形式的目的地 1.4 JMS和AMQP对比 1.5 应用场景 1.6 Spring支持 1.7 SpringBoot自动配置 1.7 市面上的MQ产品 2 RabbitMQ 2.1 RabbitMQ简介 2.1.1 RabbitMQ简介 2…

什么是Elasticsearch SQL

什么是Elasticsearch SQL 一. 介绍二. SQL 入门 前言 这是我在这个网站整理的笔记,有错误的地方请指出&#xff0c;关注我&#xff0c;接下来还会持续更新。 作者&#xff1a;神的孩子都在歌唱 一. 介绍 Elasticsearch SQL 是一个 X-Pack 组件&#xff0c;允许针对 Elasticsea…

通俗易懂理解G-GhostNet轻量级神经网络模型

一、参考资料 原始论文&#xff1a;[1] IJCV22 | 已开源 | 华为GhostNet再升级&#xff0c;全系列硬件上最优极简AI网络 二、G-GhostNet相关介绍 G-GhostNet 又称为 GhostNetV1 的升级版&#xff0c;是针对GPU优化的轻量级神经网络。 1. 摘要 GhostNetV1 作为近年来最流行…

Leetcode 611.有效三角形的个数

题目 给定一个包含非负整数的数组 nums &#xff0c;返回其中可以组成三角形三条边的三元组个数。 示例 1: 输入: nums [2,2,3,4] 输出: 3 解释:有效的组合是: 2,3,4 (使用第一个 2) 2,3,4 (使用第二个 2) 2,2,3示例 2: 输入: nums [4,2,3,4] 输出: 4提示: 1 < nums…

Android的LiveData

LiveData 是一种可观察的数据存储器类。与常规的可观察类不同&#xff0c;LiveData 具有生命周期感知能力&#xff0c;意指它遵循其他应用组件&#xff08;如 activity、fragment 或 service&#xff09;的生命周期。这种感知能力可确保 LiveData 仅更新处于活跃生命周期状态的…

ChatGPT在医学领域的应用与前景

标题&#xff1a; ChatGPT在医学领域的应用与前景 正文&#xff1a; 随着人工智能技术的不断进步&#xff0c;ChatGPT等语言模型在医学领域的应用逐渐深入&#xff0c;展现出其巨大的潜力和广阔的发展前景。作为一个高级的自然语言处理工具&#xff0c;ChatGPT能够理解和生成…

WPF 开发调试比较:Visual Studio 原生和Snoop调试控制台

文章目录 前言运行环境简单的WPF代码实现一个简单的ListBoxVisual Studio自带代码调试热重置功能测试实时可视化树查找窗口元素显示属性 Snoop调试使用Snoop简单使用调试控制台元素追踪结构树Visual/可视化结构树Logical/本地代码可视化树AutoMation/自动识别结构树 WPF元素控制…

基于springboot+vue的房屋租赁管理系统(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

【OpenAI官方课程】第四课:ChatGPT文本推断Summarizing

欢迎来到ChatGPT 开发人员提示工程课程&#xff08;ChatGPT Prompt Engineering for Developers&#xff09;&#xff01;本课程将教您如何通过OpenAI API有效地利用大型语言模型&#xff08;LLM&#xff09;来创建强大的应用程序。 本课程由OpenAI 的Isa Fulford和 DeepLearn…

手拉手Vite+Vue3+TinyVue+Echarts+TailwindCSS

技术栈springboot3hutool-alloshi-coreVue3viteTinyVueEchartsTailwindCSS软件版本IDEAIntelliJ IDEA 2022.2.1JDK17Spring Boot3.1hutool-all5.8.18oshi-core6.4.1Vue35.0.10vite5.0.10axios1.6.7echarts5.4.3 ECharts是一个使用 JavaScript 实现的开源可视化库&#xff0c;可…

快速搭建ARM64实验平台(QEMU虚拟机+Debian)

文章目录 前言一、实验平台介绍二、安装步骤2.1 安装工具2.2 下载仓库2.3 编译内核并制作根文件系统2.4 运行刚才编译好的ARM64版本的Debian系统2.5 在线安装软件包2.6 在QEMU虚拟机和主机之间共享文件 三、单步调试ARM64 Linux内核参考资料 前言 最近翻阅笨叔的《奔跑吧Linux…

go-zero微服务入门教程

go-zero微服务入门教程 本教程主要模拟实现用户注册和用户信息查询两个接口。 准备工作 安装基础环境 安装etcd&#xff0c; mysql&#xff0c;redis&#xff0c;建议采用docker安装。 MySQL安装好之后&#xff0c;新建数据库dsms_admin&#xff0c;并新建表sys_user&#…

【Git】 删除远程分支

Git 删除远程分支有以下几种方法 服务端UI工具 Git 的服务端图形化工具主要是 web 端。常用的有 GitHub、Gitea、Gutlab 等。 这些工具都提供了分支管理&#xff0c;可以直接在各服务端找到相关功能&#xff0c;谨慎删除。 客户端UI工具 Git 拥有诸多客户端 UI 工具&#x…

详细分析Python中的unittest测试框架

目录 1. 基本知识2. API2.1 断言2.2 setUp() 和 tearDown() 3. Demo 1. 基本知识 unittest 是 Python 标准库中的一个单元测试框架&#xff0c;用于编写和执行测试用例以验证代码的正确性 提供了一种结构化的方法来编写测试&#xff0c;使得测试代码更加模块化和易于维护 以…

【ACW 服务端】页面操作Java增删改查代码生成

版本: 1.2.2-JDK17-SNAPSHOT 项目地址&#xff1a;wu-smart-acw 演示地址&#xff1a;演示地址 admin/admin Java增删改查代码生成 找到对应菜单 选择你需要的数据实例 选择数据库 选择数据库表 选择客户端&#xff08;如果是本地ACW服务代码启动默认注册上的客户端ID是…

腾讯云主机Ubuntu22.04安装Odoo17

一、安装PostgreSQL16 参见之前的文章 Ubuntu22.04安装PostgreSQL-CSDN博客 二、安装Odoo17 本方案使用的nightly版的odoo&#xff0c;安装的都是最新版odoo wget -O - https://nightly.odoo.com/odoo.key | apt-key add - echo "deb http://nightly.odoo.com/17.0/n…

Maven【1】(命令行操作)

文章目录 一丶创建maven工程二、理解pom.xml三、maven的构建命令1.编译操作2.清理操作3.测试操作4.打包操作5.安装操作 一丶创建maven工程 首先创建这样一个目录&#xff0c;然后从命令行里进入这个目录&#xff1a; 然后接下来就在这个命令行里进行操作了。 这个命令是&…

Python学习笔记——PySide6设计GUI应用之UI与逻辑分离

1、打开PySide6的UI设计工具pyside6-designer&#xff0c;设计一个主窗口&#xff0c;保存文件名为testwindow.ui 2、使用PySide6的RCC工具把testwindow.ui文件转换为testwindow_rc.py文件&#xff0c;此文件中有一个类Ui_MainWindow&#xff08;包含各种控件对象&#xff09;…

设计模式浅析(八) ·外观模式

设计模式浅析(八) 外观模式 日常叨逼叨 java设计模式浅析&#xff0c;如果觉得对你有帮助&#xff0c;记得一键三连&#xff0c;谢谢各位观众老爷&#x1f601;&#x1f601; 外观模式 概念 外观模式&#xff08;Facade Pattern&#xff09;是一种设计模式&#xff0c;它为…