0

0

如何用智能指针管理第三方库资源 自定义删除器处理C风格资源释放

P粉602998670

P粉602998670

发布时间:2025-07-08 10:50:02

|

897人浏览过

|

来源于php中文网

原创

需要自定义删除器是因为智能指针默认使用delete释放资源,而第三方库资源如file*需用fclose等特定方式释放。1. unique_ptr通过模板参数添加删除器,如用fclose关闭文件:auto file = std::unique_ptr(fopen("example.txt", "r"), &fclose); 2. shared_ptr通过构造函数指定删除器,如用dlclose关闭动态库:auto handle = std::shared_ptr(dlopen("libsomething.so", rtld_lazy), dlclose); 3. 注意事项包括函数签名匹配、const正确性、lambda作删除器及跨平台差异处理。

如何用智能指针管理第三方库资源 自定义删除器处理C风格资源释放

在使用 C++ 编程时,智能指针(如 std::unique_ptrstd::shared_ptr)是管理动态内存的首选方式。但当你面对第三方库提供的资源(比如 FILE*、C 风格对象、句柄等),这些资源往往不能直接用默认的删除器释放,这时候就需要自定义删除器来配合智能指针一起使用。

如何用智能指针管理第三方库资源 自定义删除器处理C风格资源释放

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

智能指针的核心优势在于自动释放资源,避免内存泄漏。不过它们默认使用的删除方式是 deletedelete[],适用于普通的 new 出来的对象。但如果你用的是第三方库,比如打开文件用的是 fopen(),得到的是一个 FILE*,那关闭它应该调用 fclose(),而不是 delete。这个时候,默认的删除逻辑就不合适了。

如何用智能指针管理第三方库资源 自定义删除器处理C风格资源释放

这时候,你就需要通过自定义删除器告诉智能指针:“当不再需要这个资源的时候,请你用我指定的方式去释放。”


如何为 unique_ptr 添加自定义删除器?

std::unique_ptr 支持带删除器的模板参数,使用起来也比较简单。以 FILE* 为例:

如何用智能指针管理第三方库资源 自定义删除器处理C风格资源释放
auto file = std::unique_ptr(
    fopen("example.txt", "r"), 
    &fclose
);

这段代码做了几件事:

  • 使用 fopen 打开文件,返回一个 FILE*
  • 传入 fclose 作为删除器
  • file 离开作用域时,会自动调用 fclose 关闭文件
注意:这里使用了 decltype(&fclose) 来推导删除器函数的类型,这样编译器才知道怎么处理。

这种方式也适用于其他类似资源,比如网络连接句柄、图形上下文等等。


shared_ptr 的自定义删除器怎么用?

如果你希望多个智能指针共享同一个资源,并在最后一个引用离开作用域时才释放资源,可以用 std::shared_ptr。它的自定义删除器写法稍微不同:

FreeTTS
FreeTTS

FreeTTS是一个免费开源的在线文本到语音生成解决方案,可以将文本转换成MP3,

下载
auto handle = std::shared_ptr(dlopen("libsomething.so", RTLD_LAZY), dlclose);

上面的例子中:

  • 使用 dlopen 打开动态库,返回一个 void*
  • 删除器是 dlclose
  • 每次复制 handle 都会增加引用计数,只有最后一个实例销毁时才会真正调用 dlclose

小贴士:shared_ptr 的删除器是在控制块中保存的,所以即使你换了不同的 deleter 类型,只要最终释放逻辑一致就没问题。


常见坑点和注意事项

  • 函数签名要匹配:删除器函数必须接受一个指向资源类型的指针。例如,fclose 接收的是 FILE*,那你的智能指针类型也应该是 unique_ptr

  • 不要忘记 const 正确性:有些库函数可能要求非 const 指针,而智能指针默认允许修改内容,这通常没问题,但要注意是否涉及线程安全。

  • lambda 表达式也可以用作删除器:比如你想做一些额外清理动作,可以写成这样:

    auto res = std::unique_ptr(
        create_resource(), 
        [](MyResource* r) { cleanup(r); }
    );
  • 小心跨平台差异:比如 Windows 上的 HANDLE 和 Linux 上的文件描述符,它们的关闭方式不同,记得根据平台选择正确的删除逻辑。


基本上就这些。用好自定义删除器,能让你更安全地集成第三方库资源,减少手动释放带来的风险。虽然看起来有点绕,但一旦熟悉了结构,其实不复杂但容易忽略。

相关专题

更多
fclose函数的用法
fclose函数的用法

fclose是一个C语言和C++中的标准库函数,用于关闭一个已经打开的文件,是文件操作中非常重要的一个函数,用于将文件流与底层文件系统分离,释放相关的资源。更多关于fclose函数的相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

329

2023.11.30

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

527

2023.09.20

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

98

2025.11.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

205

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

50

2026.01.05

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

482

2023.08.10

菜鸟裹裹入口以及教程汇总
菜鸟裹裹入口以及教程汇总

本专题整合了菜鸟裹裹入口地址及教程分享,阅读专题下面的文章了解更多详细内容。

0

2026.01.22

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.6万人学习

Git 教程
Git 教程

共21课时 | 2.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号