1、为什么需要weak_ptr?

在正式介绍weak_ptr之前,我们先来回忆一下shared_ptr的一些知识。我们知道shared_ptr是采用引用计数的智能指针,多个shared_ptr实例可以指向同一个动态对象,并维护了一个共享的引用计数器。对于引用计数法实现的计数,总是避免不了循环引用(或环形引用)的问题,shared_ptr也不例外。

我们先来看看下面这个例子:

#include <iostream>
#include <memory>
#include <vector>
using namespace std;class ClassB;class ClassA
{
public:ClassA() { cout << "ClassA Constructor..." << endl; }~ClassA() { cout << "ClassA Destructor..." << endl; }shared_ptr<ClassB> pb;  // 在A中引用B
};class ClassB
{
public:ClassB() { cout << "ClassB Constructor..." << endl; }~ClassB() { cout << "ClassB Destructor..." << endl; }shared_ptr<ClassA> pa;  // 在B中引用A
};int main()
{shared_ptr<ClassA> spa = make_shared<ClassA>();shared_ptr<ClassB> spb = make_shared<ClassB>();spa->pb = spb;spb->pa = spa;
    std::cout << "spa use_cout:" << spa.use_count() << " spb use_cout:" << spb.use_count() << std::endl;  //spa: 2 spb:2
  // 函数结束,思考一下:spa和spb会释放资源么? 超过作用于时引用计数减一,此时为2,减一后不为0,所以内存不释放
}

上面代码的输出如下:

ClassA Constructor...
ClassB Constructor...
Program ended with exit code: 0

从上面代码中,ClassA和ClassB间存在着循环引用,从运行结果中我们可以看到:当main函数运行结束后,spa和spb管理的动态资源并没有得到释放,产生了内存泄露。为了解决类似这样的问题,C++11引入了weak_ptr,来打破这种循环引用。

2、weak_ptr是什么?

weak_ptr是为了配合shared_ptr而引入的一种智能指针,它指向一个由shared_ptr管理的对象而不影响所指对象的生命周期,也就是将一个weak_ptr绑定到一个shared_ptr不会改变shared_ptr的引用计数。不论是否有weak_ptr指向,一旦最后一个指向对象的shared_ptr被销毁,对象就会被释放。从这个角度看,weak_ptr更像是shared_ptr的一个助手而不是智能指针。

3、weak_ptr如何使用?

接下来,我们来看看weak_ptr的简单用法。

3.1如何创建weak_ptr实例

当我们创建一个weak_ptr时,需要用一个shared_ptr实例来初始化weak_ptr,由于是弱共享,weak_ptr的创建并不会影响shared_ptr的引用计数值。

示例:

    {std::shared_ptr<int> fsPtr(new int(5));std::weak_ptr<int> fwPtr = fsPtr;//weak_ptr不会改变shared_ptr,但是会和shared_ptr的引用保持一致std::cout << "fsPtr use_count:" << fsPtr.use_count() << " fwPtr use_count:" << fwPtr.use_count() << std::endl;//fwPtr.lock()后会该变shared_ptr的引用计数(+1)//std::shared_ptr<int> fsPtr2 = fwPtr.lock();//std::cout << "fsPtr use_count:" << fsPtr.use_count() << " fwPtr use_count:" << fwPtr.use_count() << std::endl;//编译报错,weak_ptr没有重载*,->操作符,因此不可直接通过weak_ptr使用对象,只能通过lock()使用shared_ptr来操作//std::cout << " number is " << *fwPtr << std::endl;fsPtr.reset();if (fwPtr.expired()){std::cout << "shared_ptr object has been destory" << std::endl;}std::shared_ptr<int> fsPtr3 = fwPtr.lock();                //fsPtr3为NULLstd::cout << " number is " << *fsPtr3 << std::endl;     //运行时中断}

3.2如何判断weak_ptr指向对象是否存在

既然weak_ptr并不改变其所共享的shared_ptr实例的引用计数,那就可能存在weak_ptr指向的对象被释放掉这种情况。这时,我们就不能使用weak_ptr直接访问对象。那么我们如何判断weak_ptr指向对象是否存在呢?C++中提供了lock函数来实现该功能。如果对象存在,lock()函数返回一个指向共享对象的shared_ptr,否则返回一个空shared_ptr。

示例:

class A
{
public:A() : a(3) { cout << "A Constructor..." << endl; }~A() { cout << "A Destructor..." << endl; }int a;
};int main() {shared_ptr<A> sp(new A());weak_ptr<A> wp(sp);//sp.reset();if (shared_ptr<A> pa = wp.lock()){cout << pa->a << endl;}else{cout << "wp指向对象为空" << endl;}
}

试试把sp.reset()这行的注释去掉看看结果有什么不同。

除此之外,weak_ptr还提供了expired()函数来判断所指对象是否已经被销毁。

示例:

class A
{
public:A() : a(3) { cout << "A Constructor..." << endl; }~A() { cout << "A Destructor..." << endl; }int a;
};int main() {shared_ptr<A> sp(new A());weak_ptr<A> wp(sp);sp.reset(); // 此时sp被销毁cout << wp.expired() << endl;  // true表示已被销毁,否则为false
}

代码输入如下:

A Constructor...
A Destructor...

3.3如何使用weak_ptr

weak_ptr并没有重载operator->和operator *操作符,因此不可直接通过weak_ptr使用对象,典型的用法是调用其lock函数来获得shared_ptr示例,进而访问原始对象。

最后,我们来看看如何使用weak_ptr来改造最前面的代码,打破循环引用问题。

class ClassB;class ClassA
{
public:ClassA() { cout << "ClassA Constructor..." << endl; }~ClassA() { cout << "ClassA Destructor..." << endl; }weak_ptr<ClassB> pb;  // 在A中引用B
};class ClassB
{
public:ClassB() { cout << "ClassB Constructor..." << endl; }~ClassB() { cout << "ClassB Destructor..." << endl; }weak_ptr<ClassA> pa;  // 在B中引用A
};int main() {shared_ptr<ClassA> spa = make_shared<ClassA>();shared_ptr<ClassB> spb = make_shared<ClassB>();spa->pb = spb;spb->pa = spa;// 函数结束,思考一下:spa和spb会释放资源么?因为没改变shared_ptr的引用计数,此时引用计数为1,超过作用域后自动释放
}

输出结果如下:

ClassA Constructor...
ClassB Constructor...
ClassA Destructor...
ClassB Destructor...
Program ended with exit code: 0

从运行结果可以看到spa和spb指向的对象都得到释放!

C++ 11 创建和使用共享 weak_ptr相关推荐

  1. xen服务器池虚拟机自启,给在XenCenter管理的XenServer服务器上创建的虚拟机共享(上传)宿主机的镜像文件...

    以win10为例,首先将要上传的镜像文件放在一个文件夹里,然后设置文件夹为共享文件夹.此处新建一个用户.步骤为"控制面板"-"用户账户"-"管理其他账 ...

  2. 在Win8中创建热点,共享网络

    在Win8中创建热点,共享网络 办公室中,我独享10M光纤,没什么要下的,便想利用来更新下Ipad里面的程序,下点公开课. 那在不利用软件[用很多wifi共享的软件],从win7开始 系统本身就自带相 ...

  3. 企业微信如何创建微盘共享空间?

    企业微信的微盘可以设置共享空间,将文件上传到共享空间后可以分享给所有空间人员,那么企业微信如何创建微盘共享空间呢? 前言 当企业使用企业微信移动办公时,如果由文件需要分享给同事,可以使用企业微信的微盘 ...

  4. django1.11 mysql配置_使用Django1.11创建简单的资产管理平台

    使用Django1.11创建简单的资产管理平台 发布时间:2020-06-05 00:06:17 来源:51CTO 阅读:883 作者:运维小学生 1:首先创建一个django项目[root@loca ...

  5. iphone创建无线网络连接服务器,iphone如何创建个人热点共享WIFI网络

    手机个人热点就好比一个路由器,它可以把你的数据网络共享给其他人使用.当你无线笔记本或别人的手机无法上网的时候,你可以把你的手机热点开启,让别人连接你的手机网络上网. 下面我们就看看如何设置iphone ...

  6. C++11学习之share_ptr和weak_ptr

    因为最近公司项目需要使用C++11,本文是在我学习share_ptr和weak_ptr过程中整理网上的资料进行写的.如有错误之处,还望小伙伴们踊跃留言.  参考资料:  http://blog.csd ...

  7. [C++11]弱引用智能指针weak_ptr初始化和相关的操作函数

    弱引用智能指针 std::weak_ptr 可以看做是 shared_ptr 的助手,它不管理 shared_ptr 内部的指针.std::weak_ptr 没有重载操作符 * 和 ->,因为它 ...

  8. C++ 11 创建和使用 shared_ptr

    shared_ptr 的类型是C + +标准库中一个聪明的指针,是为多个拥有者管理内存中对象的生命周期而设计的.在你初始化一个 shared_ptr 后,你可以复制它,把函数参数的值递给它,并把它分配 ...

  9. Linux0.11 创建进程的过程分析--fork函数的使用

    /* * linux/kernel/fork.c * * (C) 1991 Linus Torvalds */ /* 注意:signal.c和fork.c文件的编译选项内不能有vc变量优化选项/Og, ...

最新文章

  1. 如何借助 svn update 自动更新线上的web
  2. 【数字信号处理】相关函数 ( 卷积与交换性 | 相关函数不具有交换性 | 推导过程 )
  3. Evaluation of Deep Learning Toolkits
  4. idea代码提示插件_IDEA 插件推荐 —— 让你写出好代码的神器!
  5. naarray查询 swift_Swift 4最全的新特性详细解析(推荐)
  6. python学习笔记(十五)标准库StringIO和BytesIO
  7. Asterisk使用数据库配置方法
  8. 转债---Pregel: A System for Large-Scale Graph Processing(译)
  9. python表格写操作单元格合并
  10. iis如何处理并发请求
  11. 亚马逊EC2根硬盘空间扩容
  12. 【LINQ】Linq to SQL -- Select语句
  13. 基于SpringBoot的后台管理系统(启动类解析,开源的世界真好)(一)
  14. HDU——T 1573 X问题
  15. java 格式化 浮点数_DecimalFormat的用法 Java 浮点数 Float Double 小数 格式化 保留小数位后几位等...
  16. 我从可用性测试中学到的五件事
  17. HTTP协议状态码大全
  18. 计算机232接口接线,9针rs232串口接线图以及接线方法
  19. App、小程序、H5的区别
  20. 解决Redis持久化数据丢失

热门文章

  1. hdu1257(最少拦截系统 )
  2. (转)AS3 Socket
  3. HarmonyOS之AI能力·IM类意图识别
  4. Python视频编辑库:MoviePy
  5. 中国大学MOOC 人工智能导论第六章测试
  6. 经典数学问题:Nim游戏
  7. 2017/Province_Java_B/4/魔方状态
  8. ROS通信架构(上)
  9. 【STM32】SysTick定时器
  10. 【ARM】Tiny4412裸板编程之异常(软中断)