C++11的std::shared_ptr 是否是线程安全

结论:指针和引用计数是线程安全的,但指针所指对象中的操作就需要自己做控制,并不是线程安全的。因为shared_ptr 有两个数据成员(指向被管理对象的指针,和指向控制块的指针),读写操作不能原子化。

std::shared_ptr大概总结有以下几点:

(1) 智能指针主要的用途就是方便资源的管理,自动释放没有指针引用的资源。

(2) 使用引用计数来标识是否有多余指针指向该资源。(注意,shart_ptr本身指针会占1个引用)

(3) 在赋值操作中, 原来资源的引用计数会减一,新指向的资源引用计数会加一。

(4) 引用计数加一/减一操作是原子性的,所以线程安全的。

(5) make_shared要优于使用new,make_shared可以一次将需要内存分配好。

(6)std::shared_ptr的大小是原始指针的两倍,因为它的内部有一个原始指针指向资源,同时有个指针指向引用计数。

(7) 引用计数是分配在动态分配的,std::shared_ptr支持拷贝,新的指针获可以获取前引用计数个数。

下面是一段示例代码,注释详细:

std::shared_ptr<Test> p = std::make_shared<Test>();
std::shared_ptr<Test> p(new Test);

std::shared_ptr<Test> p1(new Test);
std::shared_ptr<Test> p2(new Test);
p1 = p2;
include <iostream>
#include <memory>
#include <thread>
#include <chrono>
#include <mutex>

struct Test
{
 Test() { std::cout << " Test::Test()\n"; }
 ~Test() { std::cout << " Test::~Test()\n"; }
};

//线程函数
void thr(std::shared_ptr<Test> p)
{
 //线程暂停1s
 std::this_thread::sleep_for(std::chrono::seconds(1));

 //赋值操作, shared_ptr引用计数use_cont加1(c++11中是原子操作)
 std::shared_ptr<Test> lp = p;
 {
 //static变量(单例模式),多线程同步用
 static std::mutex io_mutex;

 //std::lock_guard加锁
 std::lock_guard<std::mutex> lk(io_mutex);
 std::cout << "local pointer in a thread:\n"
 << " lp.get() = " << lp.get()
 << ", lp.use_count() = " << lp.use_count() << '\n';
 }
}

int main()
{
 //使用make_shared一次分配好需要内存
 std::shared_ptr<Test> p = std::make_shared<Test>();
 //std::shared_ptr<Test> p(new Test);

 std::cout << "Created a shared Test\n"
 << " p.get() = " << p.get()
 << ", p.use_count() = " << p.use_count() << '\n';

 //创建三个线程,t1,t2,t3
 //形参作为拷贝, 引用计数也会加1
 std::thread t1(thr, p), t2(thr, p), t3(thr, p);
 std::cout << "Shared ownership between 3 threads and released\n"
 << "ownership from main:\n"
 << " p.get() = " << p.get()
 << ", p.use_count() = " << p.use_count() << '\n';
 //等待结束
 t1.join(); t2.join(); t3.join();
 std::cout << "All threads completed, the last one deleted\n";

 return 0;
}

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值