多线程读写 shared_ptr需要加锁的原因2014-04-03 陈硕 我在《Linux 多线程服务端编程:使用 muduo C++ 网络库》第 1.9 节“再论 shared_ptr 的线程 安全”中写道:(shared_ptr)的引用计数本身是安全且无锁的,但对象的读写则不是,因为 shared_ptr 有两个数据成员,读写操作不能原子化。根据文档 (http://www.boost.org/doc/libs/release/libs/smart_ptr/shared_ptr.htm#ThreadSafety), shared_ptr 的线程安全级别和内建类型、标准库容器、std::string 一样,即:一个 shared_ptr 对象实体可被多个线程同时读取(文档例1);两个 shared_ptr 对象实体可以 被两个线程同时写入(例2),“析构”算写操作;如果要从多个线程读写同一个 shared_ptr 对象,那么需要加锁(例3~5)。请注意,以上是 shared_ptr 对象本身的线程安 全级别,不是它管理的对象的线程安全级别。后文(p.18)则介绍如何高效地加锁解锁。本文 则具体分析一下为什么“因为 shared_ptr 有两个数据成员,读写操作不能原子化”使得多线程读写同 一个 shared_ptr 对象需要加锁。这个在我看来显而易见的结论似乎也有人抱有疑问,那将导致灾难性 的后果,值得我写这篇文章。本文以 boost::shared_ptr 为例,与 std::shared_ptr 可能略有区别。shared_ptr 的数据结构shared_ptr 是引用计数型(reference counting)智能指针 ,几乎所有的实现都采用在堆(heap)上放个计数值(count)的办法(除此之外理论上还有用循环链 表的办法,不过没有实例)。具体来说,shared_ptr<Foo> 包含两个成员,一个是指向 Foo 的 指针 ptr,另一个是 ref_count 指针(其类型不一定是原始指针,有可能是 class 类型,但不影响这 里的讨论),指向堆上的 ref_count 对象。ref_count 对象有多个成员,具体的数据结构如图 1 所示 ,其中 deleter 和 allocator 是可选的。

图 1:shared_ptr 的数 据结构。为了简化并突出重点,后文只画出 use_count 的值:

以上是 shared_ptr<Foo> x(new Foo); 对应的内存数据结构。