0

0

如何自定义智能指针的删除器 实现文件句柄等资源的安全释放

P粉602998670

P粉602998670

发布时间:2025-07-25 16:31:01

|

874人浏览过

|

来源于php中文网

原创

自定义智能指针的删除器是为了确保非内存资源在不再需要时能被正确释放。1. 它通过raii原则将资源生命周期与对象绑定,2. 利用std::unique_ptr配合自定义删除器实现自动清理,3. 删除器可使用lambda表达式定义,4. 适用于文件句柄、网络连接等资源管理,5. 相比传统手动管理方式更安全可靠,避免异常或复杂逻辑导致的资源泄漏。

如何自定义智能指针的删除器 实现文件句柄等资源的安全释放

自定义智能指针的删除器,说白了,就是给智能指针一个“擦屁股”的指令,告诉它在资源不再需要时,除了释放内存,还要怎么妥善处理其他非内存资源,比如文件句柄、网络连接或者锁。这确保了即便程序出现异常,这些宝贵的资源也能被安全、及时地释放,避免泄漏。

如何自定义智能指针的删除器 实现文件句柄等资源的安全释放

解决方案

在C++里,要实现文件句柄这类非内存资源的安全释放,std::unique_ptr配合自定义删除器是我的首选。它的核心思想是利用RAII(Resource Acquisition Is Initialization)原则,让资源的生命周期与对象的生命周期绑定。当unique_ptr所管理的资源超出作用域时,它会自动调用你指定的删除器来清理。

来看个文件句柄的例子:

Videoleap
Videoleap

Videoleap是一个一体化的视频编辑平台

下载
如何自定义智能指针的删除器 实现文件句柄等资源的安全释放
#include    // For FILE, fopen, fclose
#include    // For std::unique_ptr
#include  // For std::cout, std::cerr

// 推荐的做法:使用Lambda表达式作为删除器
// Lambda的好处是它可以捕获上下文,并且可以直接定义在需要的地方,代码更紧凑。
auto file_closer = [](FILE* file_ptr) {
    if (file_ptr) {
        std::cout << "Using lambda deleter: Closing file. " << std::endl;
        // 实际应用中,这里应该处理 fclose 的返回值,例如记录日志
        if (fclose(file_ptr) != 0) {
            std::cerr << "Error closing file! errno: " << errno << std::endl;
        }
    }
};

// 定义一个别名,让类型声明更简洁
// std::unique_ptr 的第二个模板参数是删除器的类型,这里是 decltype(file_closer)
using UniqueFilePtr = std::unique_ptr;

void process_file(const char* filename) {
    // fopen 返回 FILE*,我们用 unique_ptr 来管理它
    // 构造时,需要把 deleter 实例也传进去
    UniqueFilePtr file(fopen(filename, "w"), file_closer);

    if (!file) {
        std::cerr << "Failed to open file: " << filename << std::endl;
        return;
    }

    // 写入一些内容
    fprintf(file.get(), "Hello from custom deleter example!\n");
    std::cout << "Successfully wrote to file: " << filename << std::endl;

    // 此时,即使这里抛出异常,或者函数提前返回,file 也会被正确关闭
    // file 在这里会自动析构,并调用 file_closer
} // file 离开作用域,file_closer 被调用

// 也可以用于其他资源,比如 WinAPI 的 HANDLE
#ifdef _WIN32
#include 
auto handle_closer = [](HANDLE h) {
    if (h && h != INVALID_HANDLE_VALUE) {
        std::cout << "Using lambda deleter: Closing WinAPI Handle." << std::endl;
        CloseHandle(h);
    }
};
// 对于 WinAPI HANDLE,通常使用 void* 作为类型,因为 HANDLE 只是一个泛型指针
using UniqueHandle = std::unique_ptr; 

void open_and_close_event() {
    // CreateEvent 返回 HANDLE
    UniqueHandle hEvent(CreateEvent(NULL, TRUE, FALSE, NULL), handle_closer);
    if (!hEvent) {
        std::cerr << "Failed to create event! Error: " << GetLastError() << std::endl;
        return;
    }
    std::cout << "Event created successfully." << std::endl;
    // hEvent 离开作用域时会自动调用 CloseHandle
}
#endif

int main() {
    process_file("test.txt");
#ifdef _WIN32
    open_and_close_event();
#endif
    std::cout << "Program finished." << std::endl;
    return 0;
}

通过这种方式,我们把资源的管理逻辑(打开、使用、关闭)封装在了一个对象里。当unique_ptr对象生命周期结束时,无论是因为正常退出、函数返回还是异常抛出,它都能保证资源被妥善清理,极大地提升了代码的健壮性和安全性。这比手动管理资源,要省心太多了。

为什么传统资源管理方式在C++中不够理想?

说实话,C++的传统资源管理方式,比如直接使用new/deletemalloc/free或者fopen/fclose,在简单场景下看起来没什么问题。但一旦代码复杂起来,尤其是涉及到异常处理、多分支逻辑或者深层函数调用时,它们就显得力不从心了,甚至可以说是个“坑”。

如何自定义智能指针的删除器 实现文件句柄等资源的安全释放

我一直觉得,手动管理资源就像在高速公路上开车,却要自己手动换挡、踩离合,还要担心发动机过热或者刹车失灵。你得时刻记住在代码的每个可能的出口(正常返回、异常抛出)都加上对应的资源释放代码。这不仅写起来麻烦,更容易遗漏。比如,你打开了一个文件,然后中间某个操作抛了异常,如果文件句柄没有被及时关闭,那就成了资源泄漏。对于内存,这可能是内存泄漏;对于文件句柄、网络套接字、互斥锁这些,就是各种系统资源的泄漏。长时间运行的程序,这些泄漏

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

158

2023.12.20

fclose函数的用法
fclose函数的用法

fclose是一个C语言和C++中的标准库函数,用于关闭一个已经打开的文件,是文件操作中非常重要的一个函数,用于将文件流与底层文件系统分离,释放相关的资源。更多关于fclose函数的相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

333

2023.11.30

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

208

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

191

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

55

2026.01.05

数据库Delete用法
数据库Delete用法

数据库Delete用法:1、删除单条记录;2、删除多条记录;3、删除所有记录;4、删除特定条件的记录。更多关于数据库Delete的内容,大家可以访问下面的文章。

275

2023.11.13

drop和delete的区别
drop和delete的区别

drop和delete的区别:1、功能与用途;2、操作对象;3、可逆性;4、空间释放;5、执行速度与效率;6、与其他命令的交互;7、影响的持久性;8、语法和执行;9、触发器与约束;10、事务处理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

213

2023.12.29

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

8

2026.01.30

c++ 字符串格式化
c++ 字符串格式化

本专题整合了c++字符串格式化用法、输出技巧、实践等等内容,阅读专题下面的文章了解更多详细内容。

8

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 8.1万人学习

Excel 教程
Excel 教程

共162课时 | 14.5万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号