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










