自定义删除器用于扩展std::shared_ptr资源管理能力,使其可正确释放数组、文件指针等特殊资源;通过函数指针、lambda或仿函数定义删除逻辑,确保delete[]、fclose等操作被调用;需注意删除器类型影响shared_ptr类型,且make_shared不支持自定义删除器。

在C++中,std::shared_ptr 不仅能自动管理动态分配对象的生命周期,还支持与自定义删除器结合使用。这在处理需要特殊释放逻辑的资源时非常有用,比如调用 fclose() 释放文件指针、使用 delete[] 释放数组、或调用第三方库的清理函数。
为什么需要自定义删除器?
默认情况下,shared_ptr 使用 delete 来释放所管理的对象。但以下场景中默认行为不够用:
- 管理数组时应使用 delete[]
- 封装 C 风格资源(如 FILE*、socket)需调用特定关闭函数
- 对象由特殊内存池分配,需调用对应释放函数
这时,自定义删除器就能确保资源被正确释放。
如何定义和使用自定义删除器
自定义删除器可以是函数指针、lambda 表达式或仿函数。它是一个可调用对象,接受原始指针作为参数。
立即学习“C++免费学习笔记(深入)”;
示例1:管理 FILE*
打开文件后用 shared_ptr 管理,避免忘记 fclose:
#include#include auto file_deleter = [](FILE* fp) { if (fp) std::fclose(fp); };
std::shared_ptr
fp(std::fopen("data.txt", "r"), file_deleter); if (fp) { // 使用文件指针读取数据 std::printf("File opened successfully.\n"); } // 离开作用域时自动调用 fclose
示例2:管理动态数组
使用 delete[] 正确释放数组内存:
auto array_deleter = [](int* ptr) {
delete[] ptr;
};
std::shared_ptr arr(new int[100], array_deleter);
arr.get()[0] = 42; // 访问元素
// 超出作用域时自动 delete[]
示例3:使用仿函数(函数对象)
适用于复杂删除逻辑或状态保持:
struct SocketDeleter {
void operator()(int sockfd) const {
if (sockfd >= 0) {
close(sockfd); // 假设是 Unix socket
}
}
};
std::shared_ptr sock(new int(socket(AF_INET, SOCK_STREAM, 0)), SocketDeleter{});
注意事项与最佳实践
使用自定义删除器时要注意以下几点:
- 删除器类型是 shared_ptr 类型的一部分,不同删除器会导致类型不同
- 避免捕获 lambda 中的大型对象,可能增加开销
- 若删除器有状态,确保其复制行为符合预期
- 优先使用 make_shared,但它不支持自定义删除器,必须直接构造 shared_ptr
基本上就这些。合理使用自定义删除器能让 shared_ptr 管理更多类型的资源,提升代码安全性和可维护性。










