C++shared_ptr与自定义删除器结合使用(自定义.删除.shared_ptr...)

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

c++shared_ptr与自定义删除器结合使用

在C++中,std::shared_ptr 不仅能自动管理动态分配对象的生命周期,还支持与自定义删除器结合使用。这在处理需要特殊释放逻辑的资源时非常有用,比如调用 fclose() 释放文件指针、使用 delete[] 释放数组、或调用第三方库的清理函数。

为什么需要自定义删除器?

默认情况下,shared_ptr 使用 delete 来释放所管理的对象。但以下场景中默认行为不够用:

  • 管理数组时应使用 delete[]
  • 封装 C 风格资源(如 FILE*、socket)需调用特定关闭函数
  • 对象由特殊内存池分配,需调用对应释放函数

这时,自定义删除器就能确保资源被正确释放。

如何定义和使用自定义删除器

自定义删除器可以是函数指针、lambda 表达式或仿函数。它是一个可调用对象,接受原始指针作为参数。

示例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">
                        <a class="aritcle_card_img" href="/ai/hyperwrite"><img src="https://img.php.cn/upload/ai_manual/001/246/273/68b6d3af694eb639.png" alt="HyperWrite"></a>
                        <div class="aritcle_card_info">
                            <a href="/ai/hyperwrite">HyperWrite</a>
                            <p>AI写作助手帮助你创作内容更自信</p>
                            <div class="">
                            <img src="/static/images/card_xiazai.png" alt="HyperWrite">
                            <span>54</span>
                            </div>
                        </div>
                        <a href="/ai/hyperwrite" class="aritcle_card_btn">
                            <span>查看详情</span>
                            <img src="/static/images/cardxiayige-3.png" alt="HyperWrite">
                        </a>
                        </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 管理更多类型的资源,提升代码安全性和可维护性。

以上就是C++shared_ptr与自定义删除器结合使用的详细内容,更多请关注知识资源分享宝库其它相关文章!

相关标签: unix c++ 作用域 为什么 red 封装 fclose Lambda 指针 delete 对象 大家都在看: c++中如何使用map_C++ map关联容器使用详解 c++中如何调用c语言函数_c++与c语言函数混合调用方法 C++环境搭建需要哪些基础步骤 C++内存模型与并发容器实现原理 如何在C++中链接一个外部库_C++外部库链接配置方法

标签:  自定义 删除 shared_ptr 

发表评论:

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。