C++中shared_ptr 是线程安全的吗?

在 C++ 中,shared_ptr 的线程安全性和实现原理可以通过以下方式通俗理解:


1. shared_ptr 是线程安全的吗?

答案:部分安全,需分场景!

  • 安全的操作

    • 引用计数的增减:多个线程同时复制或销毁指向同一对象的 shared_ptr 时,内部的引用计数是原子操作,线程安全。
    • 控制块的读写:管理引用计数的控制块(Control Block)本身是线程安全的。
  • 不安全的操作

    • 同一 shared_ptr 实例的修改:如果多个线程直接对同一个 shared_ptr 对象进行写操作(如 reset 或赋值),需要用户自行加锁。
    • 指向对象的访问shared_ptr 只管理指针的生命周期,不保证指向的对象是线程安全的!如果多个线程访问对象,仍需用户同步。

总结
✔️ 引用计数的操作是安全的。
❌ 直接操作同一 shared_ptr 对象或访问其指向的数据需用户加锁。


2. 实现原理:如何做到自动管理内存?

核心思想引用计数 + 控制块

  • 步骤解析
    1. 创建控制块:当第一个 shared_ptr 指向对象时,会创建一个控制块(包含引用计数、弱引用计数等)。
    2. 共享控制块:其他 shared_ptr 复制时,指向同一控制块,并原子地增加引用计数
    3. 销毁对象:当引用计数减到 0 时,销毁对象。
    4. 销毁控制块:当引用计数和弱引用计数均为 0 时,销毁控制块。

通俗比喻

  • 控制块像是一个“管家”,记录有多少个 shared_ptr 指向对象。
  • 每次有人(shared_ptr)加入时,管家在账本上记一笔(引用计数+1)。
  • 当有人离开时,管家划掉一笔(引用计数-1)。
  • 当账本记录清零时,管家会清理对象。

3. 引用计数存储在哪里?

答案:在堆内存的控制块中。

  • 控制块的结构

    • 引用计数use_count):记录当前有多少个 shared_ptr 指向对象。
    • 弱引用计数weak_count):记录 weak_ptr 的数量。
    • 删除器(Deleter):可选,用于自定义对象销毁方式。
    • 原始指针:指向实际管理的对象(某些实现可能优化)。
  • 控制块的创建

    • make_shared:对象和控制块在同一块堆内存中分配(更高效)。
    • 从裸指针构造:对象和控制块分开分配(多一次内存分配)。

示例

// 控制块和对象一起分配(高效)
auto p1 = std::make_shared<int>(42); // 控制块和对象分开分配
int* raw_ptr = new int(42);
std::shared_ptr<int> p2(raw_ptr);

总结对比表

特性说明
线程安全的操作引用计数的增减(原子操作)
需用户同步的操作同一 shared_ptr 实例的修改、指向对象的访问
引用计数存储位置堆内存中的控制块(由所有 shared_ptr 共享)
控制块的生命周期最后一个 shared_ptrweak_ptr 销毁后释放

代码示例:线程安全的操作

#include <memory>
#include <thread>void safe_operations() {auto ptr = std::make_shared<int>(42);// 多个线程复制 ptr(安全)std::thread t1([ptr] { /* 操作 ptr 副本 */ });std::thread t2([ptr] { /* 操作 ptr 副本 */ });t1.join(); t2.join();
}void unsafe_operations() {auto ptr = std::make_shared<int>(42);// 多个线程修改同一 ptr 实例(不安全!需加锁)std::thread t1([&] { ptr.reset(new int(10)); });std::thread t2([&] { ptr.reset(new int(20)); });t1.join(); t2.join(); // 可能导致数据竞争!
}

最终结论
shared_ptr 的引用计数是线程安全的,但直接操作同一实例或访问对象数据仍需用户同步。它的核心是通过控制块管理引用计数,控制块存储在堆内存中。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/75347.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

什么是 CSSD?

文章目录 一、什么是 CSSD&#xff1f;CSSD 的职责 二、CSSD 是如何工作的&#xff1f;三、CSSD 为什么会重启节点&#xff1f;情况一&#xff1a;网络和存储都断联&#xff08;失联&#xff09;情况二&#xff1a;收到其他节点对自己的踢出通知&#xff08;外部 fencing&#…

arm64平台下linux访问寄存器

通用寄存器 示例&#xff1a;读取寄存器值 // 用户态程序或内核代码中均可使用 unsigned long reg_value; asm volatile ("mov %0, x10" // 将X10的值保存到reg_value变量: "r" (reg_value) ); printk("X10 0x%lx\n", reg_value);示例&…

超级好用的小软件,连接电脑和手机。

将手机变成电脑摄像头的高效工具Iriun Webcam是一款多平台软件&#xff0c;能够将手机摄像头变成电脑的摄像头&#xff0c;通过简单的设置即可实现视频会议、直播、录制等功能。它支持Windows、Mac和Linux系统&#xff0c;同时兼容iOS和Android手机&#xff0c;操作简单&#x…

Mysql MIC高可用集群搭建

1、介绍 MySQL InnoDB Cluster&#xff08;MIC&#xff09;是基于 MySQL Group Replication&#xff08;MGR&#xff09;的高可用性解决方案&#xff0c;结合 MySQL Shell 和 MySQL Router&#xff0c;提供自动故障转移和读写分离功能&#xff0c;非常适合生产环境 2、部署 …

PERL开发环境搭建>>Windows,Linux,Mac OS

特点 简单 快速 perl解释器直接对源代码程序解释执行,是一个解释性的语言, 不需要编译器和链接器来运行代码>>速度快 灵活 借鉴了C/C, Basic, Pascal, awk, sed等多种语言, 定位于实用性语言,既具备了脚本语言的所有功能,也添加了高级语言功能 开源.免费 没有&qu…

ubuntu改用户权限

在 Linux 系统中&#xff0c;赋予普通用户 sudo 权限可以让他们执行一些需要 root 权限的命令&#xff0c;而不需要频繁切换到 root 用户。以下是具体步骤&#xff1a; 创建用户(useradd和adduser两种方式) 首先&#xff0c;需要创建一个新的用户。可以使用 adduser 或 usera…

蓝桥杯 web 学海无涯(axios、ecahrts)版本二

答案&#xff1a; // TODO: 待补充代码// 初始化图表的数据&#xff0c;设置周视图的初始数据 option.series[0].data [180, 274, 253, 324, 277, 240, 332, 378, 101]; // 周数据&#xff08;每周的总学习时长&#xff09; option.xAxis.data ["2月第1周", "…

Java 大视界 -- Java 大数据在智慧文旅虚拟场景构建与沉浸式体验增强中的技术支撑(168)

&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎来到 青云交的博客&#xff01;能与诸位在此相逢&#xff0c;我倍感荣幸。在这飞速更迭的时代&#xff0c;我们都渴望一方心灵净土&#xff0c;而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识&#xff0c;也…

API vs 网页抓取:获取数据的最佳方式

引言 在当今数字化时代&#xff0c;对于企业、研究人员以及开发人员而言&#xff0c;获取准确且及时的数据是大多数项目成功的关键因素。目前&#xff0c;收集网页数据主要有两种常用方法&#xff0c;即使用 API&#xff08;应用程序接口&#xff09;和网页抓取。然而&#xf…

车载以太网网络测试-25【SOME/IP-报文格式-1】

目录 1 摘要2 SOME/IP-报文格式2.1 **Service ID / 16 bits**2.2 **Method ID / Event ID / 16 bits**2.3 **Length / 32 bits**2.4 **Client ID / 16 bits**2.5 Session ID / 16 bits2.6 Protocol Version / 8 bits2.7 Interface Version / 8 bits2.8 Message Type / 8 bits2.…

Python数据可视化-第3章-图表辅助元素的定制

环境 开发工具 VSCode库的版本 numpy1.26.4 matplotlib3.10.1 ipympl0.9.7教材 本书为《Python数据可视化》一书的配套内容&#xff0c;本章为第3章-图表辅助元素的定制 本章主要介绍了图表辅助元素的定制&#xff0c;包括认识常用的辅助元素、设置坐标轴的标签、设置刻度范…

小程序30-wxml语法-声明和绑定数据

小程序页面中使用的数据均需要在Page() 方法的 data对象中进行声明定义 在将数据声明好以后&#xff0c;在 WXML 使用 Mustache 语法 ( 双大括号{{ }} ) 将变量包起来&#xff0c;从而将数据绑定 在 {{ }} 内部可以做一些简单的运算&#xff0c;支持如下几种方式: 算数运算三…

ubuntu开启黑屏现象解决

文章目录 前言一、问题描述二、解决方案1. 检查显卡驱动解决步骤&#xff1a; 2. 修复 GRUB 配置解决步骤&#xff1a; 3. 使用恢复模式解决步骤&#xff1a; 三、验证与总结 前言 在使用 Ubuntu 操作系统时&#xff0c;一些用户可能会遇到开机后屏幕黑屏的现象。这种问题可能…

Modbus TCP转Profibus DP网关接防撞雷达与PLC通讯

Modbus TCP转Profibus DP网关接防撞雷达与PLC通讯 在工业自动化领域&#xff0c;通信协议的多样性既是技术进步的体现&#xff0c;也给系统集成带来了挑战。Modbus TCP和Profibus DP是两种广泛应用于不同场景下的通信标准&#xff0c;它们各有优势但也存在着互操作性的需求。本…

分布式锁方案-Redisson

分布式锁&#xff1a;Redisson还实现了Redis文档中提到像分布式锁Lock这样的更高阶应用场景。事实上Redisson并没有不止步于此&#xff0c;在分布式锁的基础上还提供了联锁&#xff08;MultiLock&#xff09;&#xff0c;读写锁&#xff08;ReadWriteLock&#xff09;&#xff…

【AI插件开发】Notepad++ AI插件开发实践:从Dock窗口集成到功能菜单实现

一、项目背景与技术选型 在上篇文章实现"选中即问AI"功能的基础上&#xff0c;本文重点解决AI对话窗口的集成与核心功能菜单的开发。通过Notepad插件体系&#xff0c;我们将实现以下功能矩阵&#xff1a; AI交互系统&#xff1a;支持自然语言提问与任务执行代码智能…

ControlNet-Tile详解

一、模型功能与应用 1. 模型功能 ControlNet-Tile模型的主要功能是图像的细节增强和质量提升。它通过以下几个步骤实现这一目标&#xff1a; 语义分割&#xff1a;模型首先对输入的图像进行语义分割&#xff0c;识别出图像中不同的区域和对象。这一步是为了让模型理解图像的内…

英飞凌高信噪比MEMS麦克风驱动人工智能交互

导言 在英飞凌&#xff0c;我们一直坚信卓越的音频解决方案对于提升消费类设备的用户体验至关重要。我们坚定不移地致力于创新&#xff0c;在主动降噪、语音透传、录音室录音、音频变焦和其他相关技术方面取得了显著进步&#xff0c;对此我们深感自豪。作为MEMS麦克风的领先供…

【Azure】如何使用 Docker CLI 和 Compose 将容器部署到 Azure ACI

推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战在云中运行容器可能既困难又令人困惑。有太多选项可供选择,而且还需要理解从虚拟网络到安全性的不同云服务的工作方式。更不用说编排器了。…

结构化剪枝(Structured Pruning)与动态蒸馏(Dynamic Distillation)

结构化剪枝&#xff08;Structured Pruning&#xff09;技术详解 核心原理 结构化剪枝通过模块级&#xff08;如层、通道、块&#xff09;而非单个权重的方式去除冗余参数&#xff0c;保留关键子网络。其优势在于&#xff1a; 硬件友好性&#xff1a;生成规则稀疏模式&#x…