C++11智能指针之weak_ptr

前言,接上节:

为了解决循环引用,C++提供了另外一种智能指针:weak_ptr

Weak_Ptr

weak_ptr 拥有共享语义(sharing semantics)和不包含语义(not owning semantics)。这意味着,weak_ptr可以共享shared_ptr持有的资源。所以可以从一个包含资源的shared_ptr创建weak_ptr

weak_ptr不支持普通指针包含的*->操作。它并不包含资源所以也不允许程序员操作资源。既然如此,我们如何使用weak_ptr呢?

答案是从weak_ptr中创建shared_ptr然后再使用它。通过增加强引用计数,当使用时可以确保资源不会被销毁。当引用计数增加时,可以肯定的是从weak_ptr中创建的shared_ptr引用计数至少为1.否则,当你使用weak_ptr就可能发生如下问题:当shared_ptr离开作用域时,其拥有的资源会释放,从而导致了混乱。

创建

可以以shared_ptr作为参数构造weak_ptr.从shared_ptr创建一个weak_ptr增加了共享指针的弱引用计数(weak reference),意味着shared_ptr与其它的指针共享着它所拥有的资源。但是当shared_ptr离开作用域时,这个计数不作为是否释放资源的依据。换句话说,就是除非强引用计数变为0,才会释放掉指针指向的资源,在这里,弱引用计数(weak reference)不起作用。

void main( )
{
 shared_ptr<Test> sptr( new Test );
 weak_ptr<Test> wptr( sptr );
 weak_ptr<Test> wptr1 = wptr;
}

可以从下图观察shared_ptrweak_ptr的引用计数:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JYqv9al4-1592040542803)(E:\md文件资源%5CUsers%5CWGJ%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1592038334660.png)]

将一个weak_ptr赋给另一个weak_ptr会增加弱引用计数(weak reference count)。

所以,当shared_ptr离开作用域时,其内的资源释放了,这时候指向该shared_ptrweak_ptr发生了什么?weak_ptr过期了(expired)。

如何判断weak_ptr是否指向有效资源,有两种方法:

  1. 调用use-count()去获取引用计数,该方法只返回强引用计数,并不返回弱引用计数。
  2. 调用expired()方法。比调用use_count()方法速度更快。

weak_ptr调用lock()可以得到shared_ptr或者直接将weak_ptr转型为shared_ptr

void main( )
{
 shared_ptr<Test> sptr( new Test );
 weak_ptr<Test> wptr( sptr );
 shared_ptr<Test> sptr2 = wptr.lock( ); //增加强引用计数
}

如之前所述,从weak_ptr中获取shared_ptr增加强引用计数。

现在让我们见识一下weak_ptr如何解决循环引用问题:

// ConsoleApplication1.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"

using namespace std;
#include <iostream>
#include <memory>

class B;
class A
{
public:
	A() : m_a(5) { };
	~A()
	{
		cout << " A is destroyed" << endl;
	}
	void PrintSpB();
	weak_ptr<B> m_sptrB;
	int m_a;
};
class B
{
public:
	B() : m_b(10) { };
	~B()
	{
		cout << " B is destroyed" << endl;
	}
	weak_ptr<A> m_sptrA;
	int m_b;
};

void fun() {
	shared_ptr<B> sptrB(new B);
	shared_ptr<A> sptrA(new A);
	sptrB->m_sptrA = sptrA;
	sptrA->m_sptrB = sptrB;
}

int main()
{
	fun();

	getchar();
    return 0;
}


[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-a1b2vuKz-1592040542807)(E:\md文件资源%5CUsers%5CWGJ%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1592038486566.png)]

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值