weak_ptr是对对象的一种弱引用,它不会添加对象的引用计数。weak_ptr和shared_ptr之间能够相互转换。shared_ptr能够直接赋值给week_ptr,week_ptr可通过调用lock函数来获得shared_ptr(假设对象已经被释放,则返回一个空的shared_ptr)。

    单纯使用shared_ptr有时会产生问题,考虑以下的代码:

 1 class A;
2 class B;
3 typedef shared_ptr<A> A_Share;
4 typedef shared_ptr<B> B_Share;
5 class A
6 {
7 public:
8 B_Share m_b;
9 };
10
11 class B
12 {
13 public:
14 A_Share m_a;
15 };
16
17 A_Share a(new A());
18 B_Share b(new B());
19 a.m_b = b;
20 b.m_a = a;

在上面的代码中,a和b相互进行引用。在a和b离开作用域时,a和b的引用计数都是1,内存没有正常释放。

这就是所谓的循环引用。

一般来讲。解除这样的循环引用有以下有三种可行的方法:

  1. 当仅仅剩下最后一个引用的时候须要手动打破循环引用释放对象。

  2. 当parent的生存期超过children的生存期的时候。children改为使用一个普通指针指向parent。

  3. 使用弱引用的智能指针打破这样的循环引用。

尽管这三种方法都可行,但方法1和方法2都须要程序猿手动控制。麻烦且easy出错。

这里主要介绍一下第三种方法和boost中的弱引用的智能指针boost::weak_ptr。解决方法是将A和B中的随意一个类声明的变量改为week_ptr类型的。比方,改动类B后的代码例如以下:

1 class B
2 {
3 public:
4 weak_ptr<A> m_a;
5 }

改动后,b.m_a = a不会添加A对象的引用计数,因此a离开作用域时,引用计数为0。B对象的引用计数为2。在a和b离开作用域时,引用计数各减1后也为0。

强引用和弱引用

一个强引用当被引用的对象活着的话,这个引用也存在(就是说。当至少有一个强引用,那么这个对象就不能被释放)。boost::share_ptr就是强引用。

相对而言,弱引用当引用的对象活着的时候不一定存在。不过当它存在的时候的一个引用。弱引用并不改动该对象的引用计数,这意味这弱引用它并不正确对象的内存进行管理。在功能上类似于普通指针。然而一个比較大的差别是。弱引用能检測到所管理的对象是否已经被释放,从而避免訪问非法内存。

boost::weak_ptr

boost::weak_ptr<T>是boost提供的一个弱引用的智能指针,它的声明能够简化例如以下:

namespace boost {

    template<typename T> class weak_ptr {
public:
template <typename Y>
weak_ptr(const shared_ptr<Y>& r); weak_ptr(const weak_ptr& r); ~weak_ptr(); T* get() const;
bool expired() const;
shared_ptr<T> lock() const;
};
}

能够看到。boost::weak_ptr必须从一个boost::share_ptr或还有一个boost::weak_ptr转换而来,这也说明,进行该对象的内存管理的是那个强引用的boost::share_ptr。boost::weak_ptr仅仅是提供了对管理对象的一个訪问手段。

boost::weak_ptr除了对所管理对象的基本訪问功能(通过get()函数)外,还有两个经常使用的功能函数:expired()用于检測所管理的对象是否已经释放;lock()用于获取所管理的对象的强引用指针。

通过boost::weak_ptr来打破循环引用

因为弱引用不更改引用计数,类似普通指针,仅仅要把循环引用的一方使用弱引用,就可以解除循环引用。对于上面的那个样例来说,仅仅要把class B的定义进行改动就可以

最后值得一提的是,尽管通过弱引用指针能够有效的解除循环引用。但这样的方式必须在程序猿能预见会出现循环引用的情况下才干使用,也能够是说这个不过一种编译期的解决方式,假设程序在执行过程中出现了循环引用,还是会造成内存泄漏的。因此,不要觉得只要使用了智能指针便能杜绝内存泄漏。

毕竟。对于C++来说。因为没有垃圾回收机制,内存泄漏对每个程序猿来说都是一个很头痛的问题。

C++智能指针--weak_ptr的更多相关文章

  1. c/c++ 智能指针 weak_ptr 使用

    智能指针 weak_ptr 使用 weak_ptr用途: 1,解决空悬指针问题 2,解决循环引用问题 weak_ptr特点:没有*操作和->操作 weak_ptr是不控制所指对象生存周期的智能指 ...

  2. 智能指针weak_ptr记录

    智能指针weak_ptr为弱共享指针,实际上是share_ptr的辅助指针,不具备指针的功能.主要是为了协助 shared_ptr 工作,可用来观测资源的使用情况.weak_ptr 只对 shared ...

  3. Boost智能指针——weak_ptr

    循环引用: 引用计数是一种便利的内存管理机制,但它有一个很大的缺点,那就是不能管理循环引用的对象.一个简单的例子如下: #include <string>#include <iost ...

  4. 智能指针weak_ptr解决循环依赖问题

    #include <iostream> #include <memory> class Woman; class Man{ private: std::weak_ptr< ...

  5. [6] 智能指针boost::weak_ptr

    [1]boost::weak_ptr简介 boost::weak_ptr属于boost库,定义在namespace boost中,包含头文件 #include<boost/weak_ptr.hp ...

  6. weak_ptr<T>智能指针

    weak_ptr是为配合shared_ptr而引入的一种智能指针,它更像是shared_ptr的一个助手,而不是智能指针,因为它不具有普通指针的行为,没有重载operator*和operator-&g ...

  7. 智能指针shared_ptr新特性shared_from_this及weak_ptr

    enable_shared_from_this是一个模板类,定义于头文件<memory>,其原型为: template< class T > class enable_shar ...

  8. 详解 boost 库智能指针(scoped_ptr<T> 、shared_ptr<T> 、weak_ptr<T> 源码分析)

    一.boost 智能指针 智能指针是利用RAII(Resource Acquisition Is Initialization:资源获取即初始化)来管理资源.关于RAII的讨论可以参考前面的文章.在使 ...

  9. C++ | 再探智能指针(shared_ptr 与 weak_ptr)

    上篇博客我们模拟实现了 auto_ptr 智能指针,可我们说 auto_ptr 是一种有缺陷的智能指针,并且在C++11中就已经被摈弃掉了.那么本章我们就来探索 boost库和C++11中的智能指针以 ...

随机推荐

  1. win32内核程序中进程的pid,handle,eprocess之间相互转换的方法

    很有用,收下以后方便查询. 原贴地址:http://bbs.pediy.com/showthread.php?t=119193 在win32内核程序开发中,我们常常需要取得某进程的pid或句柄,或者需 ...

  2. OpenCV 例子代码的讲解、简介及库的安装 .

    转载请标明是引用于 http://blog.csdn.net/chenyujing1234 欢迎大家提出意见,一起讨论! 一.OpenCV介绍: OpenCV是由Intel性能基元(IPP)团队主持, ...

  3. 【iOS发展-53】实例探究:scrollView使用方法和解决方案无法滚动核心

    案例效果: (1)基本的就是练习scrollView的使用方法.界面里面的其它元素基本都是UIView和UIButton堆砌起来的. (2)主要用代码实现.当然,能够先用storyboard拖个scr ...

  4. Thinkpad W520 完美安装Ubuntu14.04LTS

    Thinkpad W520 完美安装Ubuntu14.04LTS Ubuntu已经升级到14.04LTS,这是个长期支持的版本号.自从上次安装12.04LTS之后一直没有升级. 于是从站点上下载Ubu ...

  5. Flexigrid折行显示问题

    上会写的Flexigrid折行显示时,获取值有问题,报错. getRows: function(){ //add by jej var rtnList = new Array(); var objRo ...

  6. 使用内容提供者和xml备份联系人

    1.通过内容提供者获取联系人信息 package com.ithaimazyh.readcontact; import java.util.ArrayList; import java.util.Li ...

  7. uboot代码2:stage2代码,启动内核

    一.uboot最终目的: 1.读出内核 do_nand read kernel { flash上存的内核:uImage = 头部 + 真正的内核; } 2.启动内核. do_bootm_linux { ...

  8. java 一个函数EnumMap返回多个值

    java 一个函数如何返回多个值   在开发过程中,经常会有这种情况,就是一个函数需要返回多个值,这是一个问题!! 网上这个问题的解决方法: 1.使用map返回值:这个方法问题是,你并不知道如何返回值 ...

  9. Windows Azure入门教学系列 (一): 创建第一个WebRole程序

    原文 Windows Azure入门教学系列 (一): 创建第一个WebRole程序 在第一篇教学中,我们将学习如何在Visual Studio 2008 SP1中创建一个WebRole程序(C#语言 ...

  10. boost::asio async_write也不能保证一次发完所有数据 二

    只有看boost源码才能弄明白发生了什么.首先我是将vector里面写入了数据,然后用boost::asio::buffer将vector构造成了mutable_buffer_1对象. 参考该文档的重 ...