首页 > 后端开发 > C++ > 正文

C++shared_ptr在多线程环境下安全使用

P粉602998670
发布: 2025-09-15 12:47:01
原创
361人浏览过
shared_ptr的引用计数线程安全,但对象访问和变量读写需外部同步。正确做法是用互斥锁保护shared_ptr变量本身,对所指对象的操作需额外加锁,避免循环引用可使用weak_ptr。

c++shared_ptr在多线程环境下安全使用

在多线程环境下使用C++的

std::shared_ptr
登录后复制
时,很多人误以为它是完全线程安全的。实际上,shared_ptr的线程安全性有明确的边界:它的引用计数操作是原子的,但对所指向对象的访问并不自动受保护。

shared_ptr本身的线程安全保证

C++标准规定:

  • 多个线程可以同时读取同一个
    shared_ptr
    登录后复制
    实例(即只读操作)是安全的。
  • 多个线程分别持有相同对象的不同
    shared_ptr
    登录后复制
    副本,各自进行拷贝或赋值,也是安全的,因为引用计数的增减是原子操作。
  • 但是,如果多个线程同时对同一个shared_ptr对象进行读写(比如一个线程拷贝,另一个线程重置),则需要外部同步机制(如互斥锁)。

简单说:引用计数安全,指针对象访问不安全,shared_ptr变量本身也不是线程安全的容器。

常见错误场景与正确做法

假设我们有一个全局的

shared_ptr<Data>
登录后复制

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

std::shared_ptr<Data> global_data;
登录后复制

错误用法:

  • 线程A执行
    global_data = std::make_shared<Data>();
    登录后复制
  • 线程B执行
    auto p = global_data;
    登录后复制

这两个操作同时修改和读取同一个

global_data
登录后复制
变量,属于数据竞争,未定义行为。

正确做法: 使用互斥锁保护对shared_ptr变量的访问:

豆包AI编程
豆包AI编程

豆包推出的AI编程助手

豆包AI编程 483
查看详情 豆包AI编程
std::mutex data_mutex;
std::shared_ptr<Data> global_data;

// 线程安全地更新
void update_data() {
    auto new_data = std::make_shared<Data>();
    std::lock_guard<std::mutex> lock(data_mutex);
    global_data = new_data;
}

// 线程安全地读取
std::shared_ptr<Data> get_data() {
    std::lock_guard<std::mutex> lock(data_mutex);
    return global_data;  // 拷贝是安全的
}
登录后复制

注意:返回的是副本,调用方拿到后可以在自己线程中使用,引用计数已增加,生命周期得到保障。

对所管理对象的线程安全需自行保证

即使多个线程各自持有一个指向同一对象的

shared_ptr
登录后复制
副本,对这个对象的读写仍需同步。

  • 例如两个线程通过各自的
    shared_ptr<Data>
    登录后复制
    调用
    data->modify()
    登录后复制
    ,若
    modify
    登录后复制
    不是线程安全的,就会出问题。
  • 解决方法:在
    Data
    登录后复制
    类内部使用互斥量,或由上层逻辑加锁。

典型模式:

class Data {
public:
    void modify() {
        std::lock_guard<std::mutex> lock(mutex_);
        // 实际修改操作
    }
private:
    mutable std::mutex mutex_;
};
登录后复制

避免循环引用,防止内存泄漏

多线程中频繁创建对象容易引入循环引用。应合理使用

std::weak_ptr
登录后复制
打破循环。

  • 比如缓存、观察者模式中,用
    weak_ptr
    登录后复制
    保存非拥有型引用。
  • 检查时先调用
    lock()
    登录后复制
    获取临时
    shared_ptr
    登录后复制
    ,确保对象存活。

示例:

std::weak_ptr<Data> cache;

auto ptr = cache.lock();
if (ptr) {
    // 安全使用ptr,引用计数已增加
    use(ptr);
} else {
    // 对象已被释放
}
登录后复制

基本上就这些。只要记住:shared_ptr帮你管生命周期,不管并发访问。变量共享要加锁,对象访问也要加锁。设计清晰,责任分明,多线程下才能安全使用。

以上就是C++shared_ptr在多线程环境下安全使用的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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