自定义删除器用于扩展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 <memory>
#include <cstdio>
<p>auto file_deleter = [](FILE* fp) {
if (fp) std::fclose(fp);
};</p><p>std::shared_ptr<FILE> fp(std::fopen("data.txt", "r"), file_deleter);</p><div class="aritcle_card flexRow">
<div class="artcardd flexRow">
<a class="aritcle_card_img" href="/ai/2049" title="CreateWise AI"><img
src="https://img.php.cn/upload/ai_manual/000/000/000/175680059668780.png" alt="CreateWise AI" onerror="this.onerror='';this.src='/static/lhimages/moren/morentu.png'" ></a>
<div class="aritcle_card_info flexColumn">
<a href="/ai/2049" title="CreateWise AI">CreateWise AI</a>
<p>为播客创作者设计的AI创作工具,AI自动去口癖、提交亮点和生成Show notes、标题等</p>
</div>
<a href="/ai/2049" title="CreateWise AI" class="aritcle_card_btn flexRow flexcenter"><b></b><span>下载</span> </a>
</div>
</div><p>if (fp) {
// 使用文件指针读取数据
std::printf("File opened successfully.\n");
}
// 离开作用域时自动调用 fclose
示例2:管理动态数组
使用 delete[] 正确释放数组内存:
auto array_deleter = [](int* ptr) {
delete[] ptr;
};
<p>std::shared_ptr<int> arr(new int[100], array_deleter);</p><p>arr.get()[0] = 42; // 访问元素
// 超出作用域时自动 delete[]
示例3:使用仿函数(函数对象)
适用于复杂删除逻辑或状态保持:
struct SocketDeleter {
void operator()(int sockfd) const {
if (sockfd >= 0) {
close(sockfd); // 假设是 Unix socket
}
}
};
<p>std::shared_ptr<int> sock(new int(socket(AF_INET, SOCK_STREAM, 0)), SocketDeleter{});
注意事项与最佳实践
使用自定义删除器时要注意以下几点:
- 删除器类型是 shared_ptr 类型的一部分,不同删除器会导致类型不同
- 避免捕获 lambda 中的大型对象,可能增加开销
- 若删除器有状态,确保其复制行为符合预期
- 优先使用 make_shared,但它不支持自定义删除器,必须直接构造 shared_ptr
基本上就这些。合理使用自定义删除器能让 shared_ptr 管理更多类型的资源,提升代码安全性和可维护性。









