C++中自定义删除器怎么用 shared_ptr等智能指针高级用法

自定义删除器在std::shared_ptr中的作用是让用户完全掌控资源销毁方式,解决非new/delete资源管理问题。1. 它允许传入函数、lambda或函数对象作为删除逻辑,确保如malloc内存、文件句柄等资源能正确释放;2. 避免new/delete不匹配导致的未定义行为;3. 支持raii机制管理c api资源,防止资源泄漏;4. 适配跨模块或数组等特殊释放需求。其核心价值在于使shared_ptr从内存管理工具升级为通用资源管理器

C++中自定义删除器怎么用 shared_ptr等智能指针高级用法

自定义删除器在

std::shared_ptr

中,本质上是让你能完全掌控

shared_ptr

所管理对象的销毁方式,不仅仅局限于默认的

delete

操作。这对于管理那些不是通过

new

在堆上分配的资源,比如文件句柄、网络套接字、C 风格的

malloc

内存,甚至是某个库内部的特殊资源,都显得尤为关键。它赋予了

shared_ptr

更大的灵活性和通用性,让智能指针的“智能”真正延伸到各种资源类型上。

C++中自定义删除器怎么用 shared_ptr等智能指针高级用法

解决方案

std::shared_ptr

允许你在构造时传入一个额外的参数,这个参数就是一个“删除器”(deleter)。这个删除器可以是一个普通的函数、一个 Lambda 表达式,或者一个函数对象(functor)。当

shared_ptr

的引用计数归零时,它会调用这个自定义的删除器来释放资源,而不是简单地执行

delete

一个非常典型的场景是管理 C 风格的内存分配,比如

malloc

出来的内存。如果直接用

new

对应的

delete

去释放

malloc

的内存,那肯定会出问题。这时候,自定义删除器就派上用场了:

立即学习“C++免费学习笔记(深入)”;

C++中自定义删除器怎么用 shared_ptr等智能指针高级用法

#include #include #include  // For fopen, fclose// 示例1: 使用lambda作为删除器,管理malloc分配的内存void example_malloc_deleter() {    std::cout << "--- 示例1: malloc内存管理 ---" << std::endl;    // 分配10个int的内存    int* data = (int*)std::malloc(sizeof(int) * 10);    if (!data) {        std::cerr << "malloc failed!" << std::endl;        return;    }    // 使用shared_ptr管理,并提供一个lambda作为删除器    // 当shared_ptr销毁时,这个lambda会被调用,执行free(data)    std::shared_ptr sp_data(data, [](int* p) {        std::cout << "Lambda deleter: Freeing malloc'd memory at " << p << std::endl;        std::free(p);    });    // 可以在这里使用sp_data...    sp_data.get()[0] = 100;    std::cout << "Data[0]: " << sp_data.get()[0] << std::endl;    // sp_data离开作用域时,lambda删除器会被调用    std::cout << "sp_data about to go out of scope." << std::endl;}// 示例2: 使用函数对象(Functor)作为删除器,管理文件句柄struct FileCloser {    void operator()(FILE* fp) {        if (fp) {            std::cout << "Functor deleter: Closing file handle " << fp << std::endl;            std::fclose(fp);        }    }};void example_file_deleter() {    std::cout << "n--- 示例2: 文件句柄管理 ---" << std::endl;    // 尝试打开一个文件    FILE* file_ptr = std::fopen("test.txt", "w");    if (!file_ptr) {        std::cerr << "Failed to open test.txt!" << std::endl;        return;    }    // 使用shared_ptr管理文件句柄,并提供FileCloser作为删除器    std::shared_ptr sp_file(file_ptr, FileCloser());    // 写入一些内容    std::fprintf(sp_file.get(), "Hello from shared_ptr!n");    std::cout << "Wrote to test.txt." << std::endl;    // sp_file离开作用域时,FileCloser::operator()会被调用    std::cout << "sp_file about to go out of scope." << std::endl;}// 示例3: 使用普通函数作为删除器void custom_array_deleter(int* arr) {    std::cout << "Function deleter: Deleting int array at " << arr << std::endl;    delete[] arr; // 注意这里是delete[],因为是new int[]分配的}void example_array_deleter() {    std::cout << "n--- 示例3: 数组内存管理 ---" << std::endl;    // new int[10]分配的数组    int* arr = new int[10];    // 使用shared_ptr管理,并提供custom_array_deleter函数作为删除器    std::shared_ptr sp_array(arr, custom_array_deleter);    sp_array.get()[0] = 200;    std::cout << "Array[0]: " << sp_array.get()[0] << std::endl;    // sp_array离开作用域时,custom_array_deleter会被调用    std::cout << "sp_array about to go out of scope." << std::endl;}int main() {    example_malloc_deleter();    example_file_deleter();    example_array_deleter();    std::cout << "nAll examples finished." << std::endl;    return 0;}

可以看到,无论是 Lambda、函数对象还是普通函数,核心思想都是一样的:提供一个可调用对象,当

shared_ptr

不再拥有资源时,它就会执行这个可调用对象来完成清理工作。选择哪种方式,通常取决于你的需求:Lambda 最简洁,适合一次性、内联的删除逻辑;函数对象适合需要维护状态或者更复杂、可复用的删除逻辑;普通函数则适合那些全局性的、无状态的清理函数。

值得一提的是,自定义删除器会作为

shared_ptr

的一部分被存储起来,这意味着

shared_ptr

的大小可能会比不带删除器时稍大一些,因为需要存储删除器的类型信息和可能的捕获状态。但这通常不是性能瓶颈,其带来的灵活性远超这点开销。

C++中自定义删除器怎么用 shared_ptr等智能指针高级用法

为什么我们需要自定义删除器?它解决了哪些常见问题?

说实话,一开始我接触自定义删除器的时候,也觉得有点绕,不就是个

delete

吗?但后来才发现它能解决多少实际问题,简直是

shared_ptr

从“智能指针”升级到“资源管理器”的关键一步。

最根本的原因是,C++ 世界里,资源的获取和释放方式是多种多样的,远不止

new

delete

这一对。想象一下,你可能从 C 库里

malloc

了一块内存,那对应的是

free

;你可能打开了一个文件句柄

FILE*

,那对应的是

fclose

;你可能获取了一个互斥锁

HANDLE

,那对应的是

CloseHandle

;甚至是你从某个工厂函数拿到一个对象,但这个对象需要通过一个特定的

release()

方法来销毁。如果

shared_ptr

只能无脑地调用

delete

,那它就无法管理这些非

new/delete

模式的资源,而这些资源在实际项目中比比皆是。

具体来说,自定义删除器解决了几个非常常见且棘手的问题:

new

delete

的不匹配:这是最直接的。比如你用了

malloc

分配内存,就不能用

delete

释放,必须用

free

。没有自定义删除器,

shared_ptr

遇到

malloc

出来的指针就会束手无策,或者说,强行使用会导致未定义行为甚至崩溃。管理 C API 返回的资源:很多 C 语言库(比如文件操作、图形库、网络库)返回的都是裸指针,这些指针需要通过特定的 C 函数来释放。例如

fopen

对应

fclose

socket

对应

closesocket

shared_ptr

加上自定义删除器,就能完美地将这些 C 风格的资源管理封装进 C++ 的 RAII 机制中,大大简化了资源生命周期的管理,避免了手动释放的遗漏。避免资源泄漏:这是 RAII(Resource Acquisition Is Initialization)的核心思想。没有自定义删除器,你可能不得不手动调用

fclose(fp)

或者

free(ptr)

。一旦代码路径复杂,比如有异常抛出,或者有多个返回点,就很容易忘记释放资源,导致泄漏。

shared_ptr

配合自定义删除器,确保了无论程序如何退出当前作用域,资源都能被正确、及时地释放。处理跨模块或跨语言边界的资源:在一些复杂的系统中,你可能从一个动态链接库(DLL/SO)或者其他语言(通过 FFI)获取资源。这些资源可能需要通过特定于该模块或语言的函数来释放。自定义删除器提供了这种桥接能力。数组的正确释放

new T[N]

分配的数组需要用

delete[]

来释放,而不是

delete

。虽然

shared_ptr

可以在 C++17 后直接支持数组,但在此之前,或者当你需要更精细控制时,自定义删除器是确保

delete[]

被调用的方式。

在我看来,自定义删除器是

shared_ptr

成为一个真正通用的“智能资源管理器”的基石。它让

shared_ptr

不再只是一个内存管理工具,而是一个能管理任何“拥有生命周期”的资源的利器。

自定义删除器的实现方式有哪些?代码示例详解

自定义删除器的实现方式主要有三种:Lambda 表达式、函数对象(Functor)和普通函数。每种方式都有其适用场景和优缺点。理解它们,就能在实际开发中灵活选择。

1. Lambda 表达式

这是现代 C++ 中最常用、最简洁的方式,尤其适合那些删除逻辑相对简单,且只在特定

shared_ptr

实例中使用的场景。Lambda 可以捕获上下文变量,这在某些需要额外信息的删除操作中非常有用。

#include #include #include void demo_lambda_deleter() {    std::cout << "n--- Lambda 表达式作为删除器 ---" << std::endl;    // 假设我们有一个从某个库获取的原始指针,需要特殊清理    // 这里用简单的new模拟,但想象它来自其他地方    std::vector* vec = new std::vector{1, 2, 3};    // 使用shared_ptr管理vec,并提供一个lambda删除器    // lambda捕获了vec,并在销毁时delete它    std::shared_ptr<std::vector> sp_vec(vec, [](std::vector* p) {        std::cout << "Lambda deleter: Deleting vector at " << p << std::endl;        delete p; // 确保是delete,而不是delete[]    });    std::cout << "Vector size: " <size() << std::endl;    // lambda也可以捕获外部变量    std::string resource_name = "My_Special_Resource";    int* raw_ptr = new int(42);    std::shared_ptr sp_raw_ptr(raw_ptr, [name = resource_name](int* p) {        std::cout << "Lambda deleter for " << name << ": Deleting int at " << p << std::endl;        delete p;    });    std::cout << "sp_raw_ptr value: " << *sp_raw_ptr << std::endl;    std::cout << "Shared pointers about to go out of scope." << std::endl;}

优点:简洁、内联、可以直接捕获上下文变量。缺点:如果删除逻辑复杂且需要复用,可能会导致代码冗余。

2. 函数对象(Functor)

函数对象是一个重载了

operator()

的类实例。它非常适合需要维护状态,或者删除逻辑比较复杂且需要在多个地方复用的场景。因为函数对象是一个类,它可以拥有成员变量来存储状态。

#include #include #include // 定义一个函数对象类class LoggerDeleter {private:    std::string log_prefix;    int counter;public:    LoggerDeleter(const std::string& prefix) : log_prefix(prefix), counter(0) {}    // 重载operator(),这就是删除器被调用的地方    template    void operator()(T* p) {        if (p) {            std::cout << log_prefix << " (Count: " << ++counter << "): Deleting object at " << p << std::endl;            delete p; // 假设是new出来的对象        }    }};void demo_functor_deleter() {    std::cout << "n--- 函数对象作为删除器 ---" << std::endl;    // 创建两个LoggerDeleter实例,每个都有自己的状态    LoggerDeleter file_logger("FILE_CLEANUP");    LoggerDeleter db_logger("DB_CONN_CLOSE");    // 使用第一个LoggerDeleter实例    std::shared_ptr sp1(new int(10), file_logger);    std::shared_ptr sp2(new double(20.5), file_logger); // 共享同一个deleter实例    // 使用第二个LoggerDeleter实例    std::shared_ptr sp3(new std::string("Hello"), db_logger);    std::cout << "Pointers created, about to go out of scope." << std::endl;    // 当sp1, sp2, sp3销毁时,各自的deleter会被调用,且可以看到counter的变化}

优点:可以维护状态,删除逻辑可复用,适用于复杂或有状态的清理操作。缺点:相比 Lambda 稍显繁琐,需要定义额外的类。

3. 普通函数

如果删除逻辑非常简单,不需要捕获任何上下文,也没有状态,并且是全局通用的,那么一个普通的 C++ 函数也可以作为删除器。

#include #include // 普通函数作为删除器void simple_int_deleter(int* p) {    std::cout << "Normal function deleter: Deleting int at " << p << std::endl;    delete p;}void demo_function_deleter() {    std::cout << "n--- 普通函数作为删除器 ---" << std::endl;    // 使用simple_int_deleter作为删除器    std::shared_ptr sp_val(new int(99), simple_int_deleter);    std::cout << "Value: " << *sp_val << std::endl;    std::cout << "Shared pointer about to go out of scope." << std::endl;}

优点:最简单直接,如果删除逻辑是无状态且通用的。缺点:无法捕获上下文,无法维护状态。

选择哪种方式,通常取决于删除逻辑的复杂性、是否需要状态以及复用程度。对于大多数情况,Lambda 表达式因其简洁性而成为首选。当需要更复杂的逻辑或状态管理时,函数对象则更有优势。

除了自定义删除器,shared_ptr还有哪些高级用法值得关注?

shared_ptr

的强大之处远不止自定义删除器。它构建了一个相当完善的智能指针生态,解决了 C++ 中资源管理和对象生命周期控制的诸多痛点。除了我们刚才详细讨论的自定义删除器,还有几个高级用法,我觉得在日常开发中特别值得我们去深入了解和应用。

1.

std::make_shared

:更安全、更高效的构造方式

你可能会习惯用

std::shared_ptr p(new T());

这样的方式来构造

shared_ptr

。但说实话,这并不是最优解。

std::make_shared()

才是推荐的做法。

它的优势在于:

效率提升

make_shared

通常只进行一次内存分配,同时为对象本身和

shared_ptr

内部的控制块(包含引用计数等信息)分配内存。而

new T()

之后再

shared_ptr(...)

,则需要两次独立的内存分配。这在性能敏感的场景下,尤其对于大量小对象的创建,差异会很明显。异常安全:在

shared_ptr p(new T(), func());

这种形式中,如果

func()

抛出异常,而

new T()

已经成功,那么

new T()

分配的内存就可能泄漏,因为

shared_ptr

还没来得及接管它。

make_shared

则避免了这种中间状态,保证了更强的异常安全性。

// 推荐auto sp1 = std::make_shared(arg1, arg2);// 不推荐(可能两次分配,异常不安全)// MyObject* raw_ptr = new MyObject(arg1, arg2); // 第一次分配// std::shared_ptr sp2(raw_ptr);      // 第二次分配(控制块)

2.

std::weak_ptr

:解决循环引用问题的利器

shared_ptr

最大的“坑”之一就是循环引用。如果对象 A 持有对象 B 的

shared_ptr

,同时对象 B 也持有对象 A 的

shared_ptr

,那么它们的引用计数永远不会归零,导致内存泄漏。

std::weak_ptr

就是为了解决这个问题而生的。它是一种“弱引用”智能指针,它不增加所指向对象的引用计数。它更像是一个观察者,可以检查它所指向的对象是否仍然存在。

以上就是C++中自定义删除器怎么用 shared_ptr等智能指针高级用法的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1469207.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
STL中的类型萃取技术如何应用 iterator_traits和type_traits实战
上一篇 2025年12月18日 17:53:44
怎样处理C++中的未定义行为 常见UB案例与规避方法
下一篇 2025年12月18日 17:54:05

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • Matplotlib 地图中多类型图例的创建与优化

    Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化

    本教程旨在解决matplotlib地图可视化中,如何在一个图例中同时展示颜色块(如区域分类)和自定义标记(如特定兴趣点)的问题。文章详细介绍了当传统`patch`对象无法正确显示标记时,如何利用`matplotlib.lines.line2d`创建标记图例句柄,并将其与颜色块图例句柄合并,从而生成一…

    2026年5月10日 用户投稿
    900
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    300
  • 利用海象运算符简化条件赋值:Python教程与最佳实践

    本文旨在探讨Python中海象运算符(:=)在条件赋值场景下的应用。通过对比传统if/else语句与海象运算符,以及条件表达式,分析海象运算符在简化代码、提高可读性方面的优势与局限性。并通过具体示例,展示如何在列表推导式等场景下合理使用海象运算符,同时强调其潜在的复杂性及替代方案,帮助开发者更好地掌…

    2026年5月10日
    300
  • Debian syslog性能优化技巧有哪些

    提升Debian系统syslog (通常基于rsyslog)性能,关键在于精简配置和高效处理日志。以下策略能有效优化日志管理,提升系统整体性能: 精简配置,高效加载: 在rsyslog配置文件中,仅加载必要的输入、输出和解析模块。 使用全局指令设置日志级别和格式,避免不必要的处理。 自定义模板: 创…

    2026年5月10日
    000
  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • RichHandler与Rich Progress集成:解决显示冲突的教程

    在使用rich库的`richhandler`进行日志输出并同时使用`progress`组件时,可能会遇到显示错乱或溢出问题。这通常是由于为`richhandler`和`progress`分别创建了独立的`console`实例导致的。解决方案是确保日志处理器和进度条组件共享同一个`console`实例…

    2026年5月10日
    300
  • 理解编程指令:当结果正确,但实现方式不符要求时

    本文探讨了在编程实践中,即使程序输出了正确的结果,但若其实现方式未能严格遵循既定指令,仍可能被视为“不正确”的问题。我们将通过具体示例,对比直接求和与累加求和两种实现策略,强调理解和遵守编程规范的重要性,以确保代码的健壮性、可维护性及符合项目要求。 在软件开发过程中,我们经常会遇到这样的情况:编写的…

    2026年5月10日
    000
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • php常量怎么用_PHP常量(define/const)定义与使用方法

    PHP中可通过define函数和const关键字定义常量,用于存储不可变值。define适用于全局作用域,支持动态名称和条件定义,如define(‘SITE_NAME’, ‘MyWebsite’);const在编译时生效,语法简洁但限制多,只能在类或全…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    300
  • 网站标题关键词更新后,搜索引擎为何仍显示旧标题?

    网站标题更新后,搜索引擎为何显示旧标题? 网站SEO优化中,站长常修改网站标题关键词,期望搜索结果显示自定义标题。然而,即使更新标签、meta keywords、meta description和结构化数据中的name属性后,搜索结果仍显示旧标题,这令人费解。本文将对此进行解释。 问题:站长修改了网…

    2026年5月10日
    300
  • c#文件怎么打开

    打开 C# 文件有三种方法:Visual Studio:启动 Visual Studio,通过“文件”菜单打开 C# 文件。文本编辑器:使用文本编辑器打开 C# 文件,将其视为普通文本。.NET Core 命令行工具:使用 csc.exe 命令行工具编译 C# 文件,生成可执行文件。 如何打开 C#…

    2026年5月10日
    300
  • 创建指定大小并填充特定数据的Golang文件教程

    本文将介绍如何使用Golang创建一个指定大小的文件,并用特定数据填充它。我们将使用 `os` 包提供的函数来创建和截断文件,从而实现快速生成大文件的目的。示例代码展示了如何创建一个10MB的文件,并将其填充为全零数据。掌握这些方法,可以方便地在例如日志系统或磁盘队列等场景中,预先创建测试文件或初始…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    400

发表回复

登录后才能评论
关注微信