自定义删除器用于适配非普通堆内存资源的释放,如文件句柄、C库资源或数组,确保正确调用fclose、delete[]等清理操作。
在C++中,std::unique_ptr 是一种独占式智能指针,用于自动管理动态分配资源的生命周期。默认情况下,它会在析构时调用 delete 释放所持有的对象。但在某些场景下,比如使用原始API(如文件句柄、C库资源)、数组内存、或需要特殊清理逻辑时,就需要为 std::unique_ptr 指定自定义删除器。
标准的 delete 不适用于所有资源类型。例如:
fopen 打开的文件需用 fclose
SDL_FreeSurface
delete[]
这时,自定义删除器就能确保资源被正确释放。
std::unique_ptr 的模板支持第二个参数——删除器类型。删除器可以是函数指针、lambda 表达式、仿函数等。
示例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, [](FI
LE 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 是不同类型,即使托管类型相同推荐使用类型别名提高可读性:
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); }
unique_ptr 的大小std::vector 或 std::array;若必须用裸指针,务必配合适当删除器std::default_delete),便于泛型编程基本上就这些。自定义删除器让 std::unique_ptr 能灵活适配各种资源管理需求,是实现RAII(获取即初始化)的关键工具之一。掌握它,能让代码更安全、清晰且不易泄漏资源。