0

0

C++如何实现基于引用计数的跨线程对象生命周期安全管理?(内存安全)

穿越時空

穿越時空

发布时间:2026-03-10 11:05:03

|

744人浏览过

|

来源于php中文网

原创

std::shared_ptr自身线程安全,但所指对象访问需额外同步;weak_ptr需在lock成功后立即使用shared_ptr访问对象,避免use-after-free;make_shared优化内存分配与缓存局部性。

c++如何实现基于引用计数的跨线程对象生命周期安全管理?(内存安全)

std::shared_ptr 在多线程中直接共享是否安全?

是安全的,但仅限于 std::shared_ptr 对象本身的拷贝/赋值/析构操作 —— 这些内部引用计数增减是原子的。真正不安全的是通过它访问所指向的对象(get()*ptr)时没有额外同步。

常见错误现象:std::shared_ptr 本身没崩溃,但业务逻辑读写对象成员时出现数据竞争、野指针或断言失败。

  • 引用计数原子操作 ≠ 所指对象线程安全
  • 多个线程同时调用 ptr->do_something(),而 do_something() 非 const 且修改成员变量 → 必须加锁或用 std::atomic 等手段保护对象状态
  • 若只读访问,且对象构造完成后不再修改(即“不可变对象”),则无需额外同步

如何避免 shared_ptr 析构时的跨线程 use-after-free?

核心在于:最后一个 std::shared_ptr 的析构可能发生在任意线程,而该析构会触发 delete(或自定义 deleter),如果此时其他线程还在用原始指针或 weak_ptr 锁定失败后继续访问,就崩了。

使用场景:工作线程持有一个 std::weak_ptr 定期尝试处理对象,主线程随时可能释放对象。

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

MemFree
MemFree

MemFree - 来自知识库和互联网的混合AI搜索,更快获取准确答案

下载
  • 永远不要从 std::weak_ptr::lock() 得到 std::shared_ptr 后,再解引用裸指针(如 ptr.get())并长期持有
  • 所有对对象的访问,必须包裹在 if (auto sp = wp.lock()) { /* 用 sp->xxx */ } 内,且不在 if 块外保留 sp.get()
  • 避免在 deleter 中执行耗时或依赖其他线程资源的操作(比如发消息、等锁),否则可能阻塞析构线程并引发死锁

为什么不能用裸 new + 自己写引用计数替代 shared_ptr?

自己实现跨线程安全的引用计数,难点不在“计数”,而在“内存序”和“析构时机”的精确控制。标准库的 std::shared_ptr 在 C++11 起已明确定义了引用计数操作的 memory_order_acq_rel 行为,且与 std::weak_ptr 协同处理 ABA 和竞态析构问题。

容易踩的坑:

  • std::atomic_int 做计数器,但忘记在递减后、delete 前插入 memory_order_acquire 栅栏 → 其他线程可能看到部分构造的对象状态
  • 在 deleter 中 delete 对象,但没确保该 delete 不会触发另一个线程正在执行的虚函数调用(尤其涉及多态和动态加载模块时)
  • 忽略 std::shared_ptr 的控制块(control block)分配策略:默认堆分配,若高频创建销毁,会成为性能瓶颈;可重载 new 或用 make_shared 减少一次分配

std::make_shared 比 new + shared_ptr 构造快在哪?

关键差异在内存布局:std::make_shared 把控制块和对象数据一次性分配在同一块内存里;而 std::shared_ptr<t>(new T)</t> 是两次独立分配 —— 控制块一次、对象一次。

性能影响:

  • 减少一次 malloc 调用,降低内存碎片和分配开销
  • 提升缓存局部性:引用计数和对象数据更可能落在同一 cache line
  • 但注意:若对象构造函数抛异常,make_shared 会同时释放控制块和未完成构造的对象;而手动 new + shared_ptr 构造时,若 new T 成功但 shared_ptr 构造失败(如控制块分配失败),会造成内存泄漏(极罕见,但需知道)

事情说清了就结束。最常被忽略的不是怎么写引用计数,而是忘了“引用计数安全”和“对象数据安全”是两层事,中间那条线划在哪,得看具体访问模式。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

846

2023.08.22

java多态详细介绍
java多态详细介绍

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

27

2025.11.27

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

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

561

2023.09.20

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

439

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

601

2023.08.10

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

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

764

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

376

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

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

27

2026.01.21

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

24

2026.03.09

热门下载

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

精品课程

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

共94课时 | 11万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21.2万人学习

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

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