自定义删除器是可调用对象,用于替代智能指针默认的资源释放方式。它在智能指针销毁时自动调用,适用于非new分配的资源或需特定清理函数的场景。unique_ptr通过模板参数指定删除器类型,如管理FILE*时传入fclose封装函数或lambda;shared_ptr则在构造时直接传递删除器,更灵活,常用于动态数组delete[]或SDL资源释放。使用时需确保删除器可复制/移动,避免循环引用,注意生命周期安全,并尽量保持轻量以保证性能。

在C++中使用智能指针时,有时默认的删除行为(如delete或delete[])无法满足需求。比如资源不是用new分配的,或者需要调用特定的清理函数(如close()、SDL_FreeSurface()等)。这时就需要自定义删除器来控制资源释放方式。
什么是自定义删除器
自定义删除器是一个可调用对象(函数指针、lambda、函数对象等),它会在智能指针生命周期结束时被自动调用,用来正确释放所管理的资源。它替代了默认的delete操作。
为 unique_ptr 自定义删除器
std::unique_ptr支持在模板参数中指定删除器类型,并在构造时传入删除器实例。
#include#include // 删除器函数 void close_file(FILE* fp) { if (fp) { std::fclose(fp); } }
// 使用 unique_ptr 管理文件 std::unique_ptr
)(FILE)> fp(std::fopen("test.txt", "r"), close_file); // 文件会在离开作用域时自动关闭
更简洁的方式:使用 lambda
立即学习“C++免费学习笔记(深入)”;
auto deleter = [](FILE* fp) { if (fp) std::fclose(fp); };
std::unique_ptr fp(std::fopen("test.txt", "r"), deleter);
技巧:使用 using 简化类型声明
using FilePtr = std::unique_ptr; // 实际中可写具体 lambda 类型或使用函数指针 using FileDeleter = void(*)(FILE*); using SafeFile = std::unique_ptr ; SafeFile fp(std::fopen("test.txt", "r"), [](FILE* f){ std::fclose(f); });
为 shared_ptr 自定义删除器
std::shared_ptr的删除器在构造时传入,且删除器类型不参与模板参数,更加灵活。
示例:管理动态数组(避免 delete[] 问题)
int* arr = new int[100]; std::shared_ptrsp(arr, [](int* p) { delete[] p; }); // 数组会在引用计数归零时自动 delete[]
示例:封装 C 风格资源释放
比如使用 SDL 图像:
SDL_Surface* surface = IMG_Load("image.png");
std::shared_ptr texture(surface, [](SDL_Surface* s) {
if (s) SDL_FreeSurface(s);
});
这样就无需手动调用SDL_FreeSurface,避免内存泄漏。
注意事项与技巧
- 删除器必须是可复制或可移动的(对于
shared_ptr) - 删除器捕获 lambda 时注意不要造成循环引用或意外持有资源
- 若删除器有状态(如记录日志),需确保其生命周期安全
- 对于
unique_ptr,删除器是类型的一部分,不同删除器类型不兼容 - 尽量使用轻量删除器,避免影响性能
基本上就这些。掌握自定义删除器,能让你的智能指针更强大、更安全地管理各种资源。











