自定义删除器用于适配非普通堆内存资源的释放,如文件句柄、C库资源或数组,确保正确调用fclose、delete[]等清理操作。

在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); }
调用方式:
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::vector或std::array;若必须用裸指针,务必配合适当删除器 - 可将删除器设为默认(如
std::default_delete),便于泛型编程
基本上就这些。自定义删除器让 std::unique_ptr 能灵活适配各种资源管理需求,是实现RAII(获取即初始化)的关键工具之一。掌握它,能让代码更安全、清晰且不易泄漏资源。











