0

0

多线程环境下如何使用shared_ptr 原子操作与线程安全保证

P粉602998670

P粉602998670

发布时间:2025-07-25 09:15:02

|

1111人浏览过

|

来源于php中文网

原创

shared_ptr在多线程环境下的核心要点是:1. shared_ptr的引用计数操作是原子且线程安全的,确保其生命周期管理不会出错;2. 但它所指向的对象内部数据并非线程安全,若对象状态在多线程中被并发修改,需额外同步机制如mutex保护;3. 可使用std::atomic>实现shared_ptr实例本身的原子替换,但这不解决对象内部数据的竞争问题;4. 推荐策略包括封装同步逻辑、设计不可变对象、避免从this创建shared_ptr、谨慎使用裸指针和weak_ptr,并优先减少共享可变状态。

多线程环境下如何使用shared_ptr 原子操作与线程安全保证

在多线程环境下使用 shared_ptr,核心要点是:shared_ptr 本身对引用计数的增减操作是原子且线程安全的,但这不意味着它所指向的那个对象内部的数据访问也是线程安全的。如果你要修改 shared_ptr 管理的对象,或者这个对象内部的状态会在多线程中被访问,那么你仍然需要额外的同步机制来保护这个对象。

多线程环境下如何使用shared_ptr 原子操作与线程安全保证

解决方案

理解 shared_ptr 在多线程中的行为,关键在于区分“管理 shared_ptr 自身的生命周期”和“管理 shared_ptr 所指向的数据”。shared_ptr 的控制块(包含引用计数和弱引用计数)的操作,如复制、赋值、销毁等,都是由 C++ 标准库保证原子性的。这意味着在多个线程同时对同一个 shared_ptr 实例进行拷贝或销毁时,引用计数不会出现竞争条件,从而避免了双重释放或提前释放的问题。

然而,这种原子性仅限于 shared_ptr 的内部管理机制。它所指向的实际数据(T类型的对象)的读写操作,如果发生在多个线程之间,并且其中至少有一个是写入操作,那么这就构成了数据竞争。为了保护这些共享数据,你需要显式地引入同步原语,比如 std::mutex、读写锁,或者设计不可变(immutable)的数据结构。

多线程环境下如何使用shared_ptr 原子操作与线程安全保证

对于 shared_ptr 实例本身的原子替换,即在一个共享变量中原子地更新 shared_ptr 指向另一个对象,可以使用 std::atomic<:shared_ptr>>。但这只解决了指针本身替换的原子性,不解决被指向对象内部数据的线程安全问题。

shared_ptr的引用计数是线程安全的吗?深入理解其内部机制

是的,shared_ptr 的引用计数操作是线程安全的。这是 C++ 标准库为 shared_ptr 设计时就明确规定的行为。当我们复制一个 shared_ptr(例如通过拷贝构造函数或赋值操作符),或者一个 shared_ptr 离开作用域被销毁时,其内部的引用计数会相应地原子增加或减少。

多线程环境下如何使用shared_ptr 原子操作与线程安全保证

具体来说,标准库的实现通常会利用底层平台的原子指令(如 fetch_addfetch_sub 或等效的锁指令)来操作引用计数。这确保了即使多个线程同时对同一个 shared_ptr 进行操作,引用计数器也能保持正确的值,从而避免了因计数错误导致的内存泄漏(引用计数永远不为零)或提前释放(引用计数过早归零导致多个 shared_ptr 访问已释放内存)。

我个人觉得,这里有个非常容易被误解的地方:很多人会因为“引用计数线程安全”就直接推断出“整个 shared_ptr 及其指向的对象都是线程安全的”,这绝对是个大坑。引用计数的安全仅仅是保证了 shared_ptr 自身的生命周期管理不出错,和它指向的那个 T 类型的对象的内部状态完全是两码事。你可以想象成 shared_ptr 是个保险箱,它自己开关锁是安全的,但保险箱里的钱(你的数据)会不会被偷走,取决于你有没有给钱再加把锁。

共享对象的数据竞争:如何正确保护shared_ptr指向的数据?

既然 shared_ptr 无法自动保护它所指向的对象,那么当多个线程需要访问或修改这个共享对象时,我们就需要主动介入。这通常是多线程编程中最核心也是最容易出错的部分。

保护 shared_ptr 指向的数据,有几种常用的策略:

  1. 使用互斥锁(std::mutex)进行同步: 这是最直接、最常见的做法。你可以在 shared_ptr 所管理的对象内部封装一个 std::mutex,或者在外部创建一个 std::mutex 来保护对该对象的访问。

    秘塔AI搜索
    秘塔AI搜索

    秘塔AI搜索,没有广告,直达结果

    下载
    • 内部封装: 推荐这种方式,因为它将数据和其保护机制紧密绑定在一起,形成一个“线程安全对象”。

      #include 
      #include 
      #include 
      #include 
      #include 
      #include 
      #include 
      
      class SharedResource {
      public:
          SharedResource(const std::string& name) : name_(name), value_(0) {
              std::cout << "Resource " << name_ << " created." << std::endl;
          }
      
          ~SharedResource() {
              std::cout << "Resource " << name_ << " destroyed." << std::endl;
          }
      
          void incrementValue() {
              std::lock_guard lock(mtx_); // 锁定互斥量
              value_++;
              std::cout << name_ << ": Value incremented to " << value_ << std::endl;
          }
      
          int getValue() const {
              std::lock_guard lock(mtx_); // 读操作也需要保护,防止读到脏数据
              return value_;
          }
      
      private:
          std::string name_;
          int value_;
          mutable std::mutex mtx_; // mutable 允许在 const 成员函数中修改
      };
      
      // 示例用法:
      // std::shared_ptr res = std::make_shared("MyData");
      // std::thread t1([&]{ for(int i=0; i<5; ++i) res->incrementValue(); });
      // std::thread t2([&]{ for(int i=0; i<5; ++i) res->incrementValue(); });
      // t1.join();
      // t2.join();
      // std::cout << "Final value: " << res->getValue() << std::endl;

      这种方式让 SharedResource 对象本身就是线程安全的,无论它是否被 shared_ptr 管理,其内部操作都能保证同步。

  2. 设计不可变(Immutable)对象: 如果你所共享的对象在创建后就不会再被修改,那么它就是天然线程安全的。shared_ptr 非常适合用来共享这样的不可变数据。这是并发编程中一种非常强大的模式,因为它完全消除了数据竞争的可能性。

    #include 
    #include 
    #include 
    #include 
    #include 
    
    class ImmutableConfig {
    public:
        ImmutableConfig(int version, const std::string& data) : version_(version), data_(data) {}
    
        int getVersion() const { return version_; }
        const std::string& getData() const { return data_; }
    
        // 没有修改成员变量的方法,因此是不可变的
    private:
        int version_;
        std::string data_;
    };
    
    // 示例用法:
    // std::shared_ptr config = std::make_shared(1, "Initial Settings");
    // std::thread t1([&]{ std::cout << "Thread 1 config version: " << config->getVersion() << std::endl; });
    // std::thread t2([&]{ std::cout << "Thread 2 config data: " << config->getData() << std::endl; });
    // t1.join();
    // t2.join();

    在这种情况下,shared_ptr 是一个很好的选择,它明确表示你无法通过这个指针修改对象。

  3. 使用 std::atomic<:shared_ptr>> 这不是用来保护 shared_ptr 指向的数据,而是用来原子地替换 shared_ptr 本身。如果你有一个 shared_ptr 变量,并且希望在多线程中原子地改变它所指向的对象(比如,更新一个全局配置指针),那么 std::atomic<:shared_ptr>> 就派上用场了。

    #include 
    #include 
    #include 
    #include 
    #include 
    
    class MyObject {
    public:
        MyObject(int id) : id_(id) { std::cout << "MyObject " << id_ << " created." << std::endl; }
        ~MyObject() { std::cout << "MyObject " << id_ << " destroyed." << std::endl; }
        int getId() const { return id_; }
    private:
        int id_;
    };
    
    std::atomic> global_object_ptr;
    
    void reader_thread() {
        for (int i = 0; i < 3; ++i) {
            std::shared_ptr current_obj = global_object_ptr.load(); // 原子加载
            if (current_obj) {
                std::cout << "Reader: Current object ID is " << current_obj->getId() << std::endl;
            } else {
                std::cout << "Reader: No object available." << std::endl;
            }
            std::this_thread::sleep_for(std::chrono::milliseconds(50));
        }
    }
    
    void writer_thread() {
        for (int i = 0; i < 2; ++i) {
            std::this_thread::sleep_for(std::chrono::milliseconds(100));
            std::shared_ptr new_obj = std::make_shared(i + 100);
            global_object_ptr.store(new_obj); // 原子存储
            std::cout << "Writer: Updated object to ID " << new_obj->getId() << std::endl;
        }
    }
    
    // 示例用法:
    // global_object_ptr.store(std::make_shared(0)); // 初始值
    // std::thread t_reader(reader_thread);
    // std::thread t_writer(writer_thread);
    // t_reader.join();
    // t_writer.join();

    这里需要强调的是,std::atomic<:shared_ptr>> 保证的是 global_object_ptr 这个变量本身的读写原子性,即保证在多线程环境下,对 global_object_ptr 进行 load()store() 操作时,不会出现撕裂(torn reads/writes)。但是,一旦你通过 load() 获得了 std::shared_ptr 的副本 current_obj,那么对 current_obj 所指向的 MyObject 内部的任何修改,仍然需要 MyObject 自身来保证线程安全。

避免常见的shared_ptr多线程陷阱与最佳实践

在多线程中使用 shared_ptr,有些坑是新手很容易踩的,甚至经验丰富的开发者也可能一时疏忽。

  1. 陷阱:误认为 shared_ptr 赋予了对象线程安全。 这大概是最常见也最危险的误解了。我前面反复强调,shared_ptr 提供的线程安全仅限于其内部的引用计数操作。它不提供对所管理对象的任何同步保证。如果你有一个 shared_ptr,并且 Foo 对象内部有成员变量会被多个线程同时读写,你必须为 Foo 的成员变量访问添加锁。

  2. 陷阱:从 this 创建 shared_ptr 在一个类成员函数内部,如果你想获取当前对象的 shared_ptr,绝不能直接 std::shared_ptr(this)。这会导致创建出第二个独立的控制块,当这两个 shared_ptr 都认为自己是最后一个持有者时,就会发生双重释放(double free)的灾难。 最佳实践: 继承 std::enable_shared_from_this

    #include 
    #include 
    
    class MyClass : public std::enable_shared_from_this {
    public:
        std::shared_ptr getSharedPtr() {
            return shared_from_this(); // 正确获取指向自身的 shared_ptr
        }
    };
    
    // std::shared_ptr obj = std::make_shared();
    // std::shared_ptr another_obj_ptr = obj->getSharedPtr(); // 安全

    记住,shared_from_this() 只有在对象已经被 shared_ptr 管理后才能安全调用。

  3. 陷阱:在 shared_ptr 生命周期之外使用其内部的裸指针。 如果你从 shared_ptr 中获取一个裸指针(例如 shared_ptr.get()),然后 shared_ptr 实例本身被销毁了(比如它是一个局部变量,函数返回了),那么它所指向的对象就会被释放。此时你手里的裸指针就成了悬空指针。在多线程环境下,这更是难以追踪的 bug。 最佳实践: 尽量直接使用 shared_ptr 实例本身,而不是频繁地 get() 裸指针。如果确实需要裸指针,确保其生命周期不会超过 shared_ptr 所管理对象的生命周期。

  4. 陷阱:weak_ptr 的误用导致竞态。weak_ptr 常常用来解决 shared_ptr 的循环引用问题。你可以从 weak_ptr 尝试 lock() 得到一个 shared_ptr。如果对象已经被销毁,lock() 会返回一个空的 shared_ptr。在多线程中,你可能会先检查 weak_ptr.expired(),然后尝试 lock()。但 expired()lock() 之间可能存在竞态,对象可能在你检查完 expired() 之后但在 lock() 之前被销毁。 最佳实践: 总是直接尝试 lock() weak_ptr,并检查返回的 shared_ptr 是否为空。

    std::shared_ptr obj_ptr = weak_obj_ptr.lock(); // 尝试锁定
    if (obj_ptr) {
        // 对象仍然存在,可以安全使用 obj_ptr
        obj_ptr->doSomething();
    } else {
        // 对象已销毁
        std::cout << "Object already expired." << std::endl;
    }
  5. 最佳实践:封装同步逻辑。 如果你的对象需要在多线程中被修改,最优雅的方式是让对象自己负责其内部数据的同步。这意味着在对象的成员函数中加入 std::mutex 或其他同步机制,而不是让外部代码来管理锁。这使得对象的使用者无需关心其内部的线程安全细节。

  6. 最佳实践:优先使用不可变数据。 如果业务逻辑允许,设计不可变的对象。一旦创建,其内部状态永不改变。这样,你就可以在多线程中自由地共享 shared_ptr,完全无需担心数据竞争。这大大简化了并发编程的复杂性。

  7. 最佳实践:最小化共享的可变状态。 这是并发编程的黄金法则。你共享的可变状态越少,你需要处理的同步问题就越少。shared_ptr 固然方便,但它也意味着你正在共享所有权。在设计系统时,多考虑如何减少这种共享的可变性。

总的来说,shared_ptr 是一个强大的工具,但它并非万能的银弹。在多线程环境中,你需要清晰地理解它所提供的保障和未提供的保障,并在此基础上,结合适当的同步机制和设计模式,才能写出健壮、高效的并发程序。

相关专题

更多
c语言const用法
c语言const用法

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

527

2023.09.20

c++怎么把double转成int
c++怎么把double转成int

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

53

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

100

2025.10.23

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

536

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

22

2026.01.06

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

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

482

2023.08.10

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

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

143

2025.12.24

c++空格相关教程合集
c++空格相关教程合集

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

0

2026.01.23

热门下载

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

精品课程

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

共58课时 | 4万人学习

Pandas 教程
Pandas 教程

共15课时 | 1.0万人学习

ASP 教程
ASP 教程

共34课时 | 3.9万人学习

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

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