首页 > 后端开发 > C++ > 正文

c++如何使用std::unique_ptr的自定义删除器_c++智能指针与资源管理扩展

php中文网
发布: 2025-12-06 21:11:02
原创
1004人浏览过
自定义删除器用于适配非普通堆内存资源的释放,如文件句柄、C库资源或数组,确保正确调用fclose、delete[]等清理操作。

c++如何使用std::unique_ptr的自定义删除器_c++智能指针与资源管理扩展

在C++中,std::unique_ptr 是一种独占式智能指针,用于自动管理动态分配资源的生命周期。默认情况下,它会在析构时调用 delete 释放所持有的对象。但在某些场景下,比如使用原始API(如文件句柄、C库资源)、数组内存、或需要特殊清理逻辑时,就需要为 std::unique_ptr 指定自定义删除器

为什么需要自定义删除器?

标准的 delete 不适用于所有资源类型。例如:

  • 通过 fopen 打开的文件需用 fclose
  • C API 返回的指针可能需要调用特定函数如 SDL_FreeSurface
  • 动态数组应使用 delete[]
  • 某些系统资源需要关闭描述符或释放非堆内存

这时,自定义删除器就能确保资源被正确释放。

如何定义和使用自定义删除器

std::unique_ptr 的模板支持第二个参数——删除器类型。删除器可以是函数指针、lambda 表达式、仿函数等。

立即学习C++免费学习笔记(深入)”;

示例1:管理 FILE* 文件流

#include <memory>
#include <cstdio>
<p>// 自定义删除器函数
void close_file(FILE* fp) {
if (fp) fclose(fp);
}</p><p>// 使用函数指针作为删除器
std::unique_ptr<FILE, void(<em>)(FILE</em>)> open_file(const char<em> name) {
return std::unique_ptr<FILE, void(</em>)(FILE*)>(fopen(name, "r"), close_file);
}
登录后复制

调用方式:

CodeGeeX
CodeGeeX

智谱AI发布的AI编程辅助工具插件,可以实现自动代码生成、代码翻译、自动编写注释以及智能问答等功能

CodeGeeX 166
查看详情 CodeGeeX
auto file = open_file("data.txt");
if (file) {
    // 使用文件...
    char buffer[256];
    fgets(buffer, sizeof(buffer), file.get());
}
// 离开作用域后自动 fclose
登录后复制

示例2:使用 lambda 表达式(更简洁)

auto deleter = [](FILE* fp) { if (fp) fclose(fp); };
std::unique_ptr<FILE, decltype(deleter)> fp(fopen("test.txt", "w"), deleter);
<p>// 或直接内联
std::unique_ptr<FILE, decltype([](FILE<em> f){if(f)fclose(f);})> fp2(nullptr, [](FILE</em> f){if(f)fclose(f);});
登录后复制

示例3:管理 C 风格数组

struct ArrayDeleter {
    void operator()(int* p) const {
        delete[] p;
    }
};
<p>std::unique_ptr<int[], ArrayDeleter> arr(new int[100], ArrayDeleter{});</p><p>// 更简单的写法:利用默认构造
std::unique_ptr<int[], void(<em>)(int</em>)> arr2(new int[100], [](int* p) { delete[] p; });
登录后复制

删除器对类型的影响

注意:当指定自定义删除器时,删除器类型会成为 unique_ptr 类型的一部分。这意味着:

  • 带有不同删除器类型的 unique_ptr 是不同类型,即使托管类型相同
  • 函数返回值必须明确写出完整类型(可用 auto 或 using 简化)
  • 空删除器(如 lambda 捕获为空)通常不增加对象体积

推荐使用类型别名提高可读性:

using FilePtr = std::unique_ptr<FILE, void(*)(FILE*)>;
using ImagePtr = std::unique_ptr<SDL_Surface, decltype(&SDL_FreeSurface)>;
<p>FilePtr open_text_file(const std::string& path) {
return FilePtr(fopen(path.c_str(), "r"), close_file);
}
登录后复制

注意事项与最佳实践

  • 若删除器无状态(如普通函数或空捕获 lambda),不会增加 unique_ptr 的大小
  • 避免在删除器中抛出异常,析构函数应安全
  • 对于数组,优先考虑 std::vectorstd::array;若必须用裸指针,务必配合适当删除器
  • 可将删除器设为默认(如 std::default_delete),便于泛型编程

基本上就这些。自定义删除器让 std::unique_ptr 能灵活适配各种资源管理需求,是实现RAII(获取即初始化)的关键工具之一。掌握它,能让代码更安全、清晰且不易泄漏资源。

以上就是c++++如何使用std::unique_ptr的自定义删除器_c++智能指针与资源管理扩展的详细内容,更多请关注php中文网其它相关文章!

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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