0

0

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

尼克

尼克

发布时间:2025-12-06 21:11:02

|

1028人浏览过

|

来源于php中文网

原创

自定义删除器用于适配非普通堆内存资源的释放,如文件句柄、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 
#include 

// 自定义删除器函数 void close_file(FILE* fp) { if (fp) fclose(fp); }

// 使用函数指针作为删除器 std::unique_ptr)(FILE)> open_file(const char name) { return std::unique_ptr)(FILE*)>(fopen(name, "r"), close_file); }

调用方式:

文心快码
文心快码

文心快码(Comate)是百度推出的一款AI辅助编程工具

下载
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 fp(fopen("test.txt", "w"), deleter);

// 或直接内联 std::unique_ptr f){if(f)fclose(f);})> fp2(nullptr, [](FILE f){if(f)fclose(f);});

示例3:管理 C 风格数组

struct ArrayDeleter {
    void operator()(int* p) const {
        delete[] p;
    }
};

std::unique_ptr arr(new int[100], ArrayDeleter{});

// 更简单的写法:利用默认构造 std::unique_ptr)(int)> arr2(new int[100], [](int* p) { delete[] p; });

删除器对类型的影响

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

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

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

using FilePtr = std::unique_ptr;
using ImagePtr = std::unique_ptr;

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(获取即初始化)的关键工具之一。掌握它,能让代码更安全、清晰且不易泄漏资源。

相关专题

更多
fclose函数的用法
fclose函数的用法

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

328

2023.11.30

lambda表达式
lambda表达式

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

204

2023.09.15

python lambda函数
python lambda函数

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

190

2025.11.08

Python lambda详解
Python lambda详解

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

49

2026.01.05

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

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

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

269

2023.11.13

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

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

210

2023.12.29

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

23

2026.01.19

热门下载

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

精品课程

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

共58课时 | 3.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

ASP 教程
ASP 教程

共34课时 | 3.7万人学习

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

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