当前位置: 首页 > news >正文

深入理解 C++20 中的 `std::shared_ptr` 原子操作

文章目录

    • 1. `std::shared_ptr` 的线程安全问题
    • 2. `std::shared_ptr` 原子操作函数
      • 2.1 原子读取和存储
      • 2.2 原子交换
      • 2.3 原子比较交换
    • 3. 注意事项
    • 4. 示例
    • 5. 总结

在多线程编程中,线程安全是一个至关重要的问题。C++11 引入了 std::shared_ptr,它通过引用计数机制提供了资源管理的便利性。然而,当多个线程需要共享和操作同一个 std::shared_ptr 对象时,线程安全问题仍然需要特别关注。幸运的是,C++ 标准库提供了针对 std::shared_ptr 的原子操作函数,这些函数可以帮助我们安全地在多线程环境中使用 std::shared_ptr

1. std::shared_ptr 的线程安全问题

std::shared_ptr 的控制块是线程安全的,这意味着不同的 std::shared_ptr 对象可以同时访问同一个控制块,而不会引发数据竞争。然而,当多个线程需要访问和修改同一个 std::shared_ptr 对象时,问题就出现了。例如,如果一个线程正在通过 resetoperator= 修改 std::shared_ptr 的指向,而另一个线程正在读取它的值,那么就可能发生数据竞争。

2. std::shared_ptr 原子操作函数

为了在多线程环境中安全地使用 std::shared_ptr,C++11 引入了一系列原子操作函数。这些函数允许我们以原子方式对 std::shared_ptr 进行读取、存储、交换和比较交换操作。

2.1 原子读取和存储

  • std::atomic_loadstd::atomic_store:这两个函数允许我们以原子方式读取和存储 std::shared_ptr 的值。它们的显式版本(std::atomic_load_explicitstd::atomic_store_explicit)还允许我们指定内存顺序。
std::shared_ptr<int> ptr = std::make_shared<int>(42);
std::shared_ptr<int> loaded_ptr = std::atomic_load(&ptr); // 原子读取
std::atomic_store(&ptr, std::make_shared<int>(100)); // 原子存储

2.2 原子交换

  • std::atomic_exchange:这个函数允许我们以原子方式交换 std::shared_ptr 的值。它会返回交换前的值。
std::shared_ptr<int> ptr = std::make_shared<int>(42);
std::shared_ptr<int> new_ptr = std::make_shared<int>(100);
std::shared_ptr<int> old_ptr = std::atomic_exchange(&ptr, new_ptr); // 原子交换

2.3 原子比较交换

  • std::atomic_compare_exchange_weakstd::atomic_compare_exchange_strong:这两个函数允许我们以原子方式进行比较交换操作。它们的显式版本(std::atomic_compare_exchange_weak_explicitstd::atomic_compare_exchange_strong_explicit)还允许我们指定成功和失败时的内存顺序。
std::shared_ptr<int> ptr = std::make_shared<int>(42);
std::shared_ptr<int> expected = ptr;
std::shared_ptr<int> desired = std::make_shared<int>(100);
if (std::atomic_compare_exchange_strong(&ptr, &expected, desired)) {
    // 交换成功
} else {
    // 交换失败
}

3. 注意事项

  • 互斥锁实现:这些原子操作函数通常使用互斥锁实现。这意味着它们可能比直接操作 std::shared_ptr 更慢,但在多线程环境中是安全的。
  • 全局哈希表:互斥锁存储在全局哈希表中,指针值用作键。这可能会导致性能问题,尤其是在高并发场景下。
  • 并发 TS:并发 TS 提供了原子智能指针类 atomic_shared_ptratomic_weak_ptr,以替代对这些函数的使用。这些类提供了更直观的语法和更好的性能。

4. 示例

以下是一个使用 std::shared_ptr 原子操作函数的简单示例:

#include <iostream>
#include <memory>
#include <thread>
#include <atomic>

void worker(std::shared_ptr<int> ptr) {
    std::shared_ptr<int> loaded_ptr = std::atomic_load(&ptr);
    std::cout << "Loaded value: " << *loaded_ptr << std::endl;
}

int main() {
    std::shared_ptr<int> ptr = std::make_shared<int>(42);
    std::thread t1(worker, std::ref(ptr));
    std::thread t2(worker, std::ref(ptr));

    t1.join();
    t2.join();

    return 0;
}

在这个示例中,两个线程同时读取 std::shared_ptr 的值。通过使用 std::atomic_load,我们可以确保读取操作是原子的,从而避免数据竞争。

5. 总结

std::shared_ptr 的原子操作函数为我们提供了一种在多线程环境中安全使用 std::shared_ptr 的方法。虽然这些函数的实现可能涉及互斥锁,从而导致性能开销,但它们可以有效避免数据竞争。在高并发场景下,建议使用并发 TS 提供的原子智能指针类,以获得更好的性能。
std::atomic_… std::shared_ptr - cppreference.cn - C++参考手册

相关文章:

  • JDK 动态代理和 CGLIB 动态代理
  • 新能源电站系统建设提速!麒麟信安操作系统驱动光伏风电双领域安全升级
  • PEFT简介
  • GitHub Copilot 在 VS Code 上的终极中文指南:从安装到高阶玩法
  • 特种兵旅游之大连3日游
  • Android视频渲染SurfaceView强制全屏与原始比例切换
  • 塔能科技:做节能界的“催化剂”,加速工厂能源改造变革
  • 「速通AI编程开发」共学(三):提示词(Prompts)配置项
  • [Spring]注解开发(2)
  • 【AutoFormer 源码理解】 conv1d
  • 【蓝桥杯】省赛:缴纳过路费(并查集)
  • 虚拟定位 1.2.0.2 | 虚拟定位,上班打卡,校园跑步模拟
  • AI幻觉时代:避坑指南与技术反思
  • 机器学习扫盲系列(2)- 深入浅出“反向传播”-1
  • 粗粒度和细粒度指的是什么?
  • 回顾Transformer,并深入讲解替代方案Mamba原理(图解)
  • 【6. 系统调用】
  • 异常(11)
  • 解决QT_Debug 调试信息不输出问题
  • Navigation页面导航的使用
  • 苹果或将于2027年推出由玻璃制成的曲面iPhone
  • 中美经贸高层会谈在瑞士日内瓦举行
  • 匈牙利外长称匈方已驱逐两名乌克兰外交官
  • 保利42.41亿元竞得上海杨浦东外滩一地块,成交楼面单价超8万元
  • 71岁导演詹姆斯・弗雷病逝,曾执导《纸牌屋》、麦当娜MV
  • 美乌基金协议:美国搞了一套可在资源富集地区复刻的商业模式