boost的shared_ptr循环引用

来源:互联网 发布:淘宝信誉查询 编辑:程序博客网 时间:2024/06/05 08:17

boost的智能指针给编程带来了极大的便利,不需要关心内存的释放,不要要调用delete,而且还可以定制delete的方法。其实boost的智能指针是可以当成scope_exit来用的,同样是退出时处理。但是凡事都是有利有弊,boost的shared_ptr如果在循环引用的时候会出现无法释放内存的情况,所谓循环引用就是A智能指针类里存放B的智能指针,B的智能指针类里存放A,将a、b的值互相设置。增加引用计数,在释放的时候由于计数问题,会导致在退出指针域的时候无法进行释放,解决该问题的方案是在类成员中使用弱指针weak_ptr

 解决std::shared_ptr循环引用问题的钥匙在weak_ptr手上。weak_ptr对象引用资源时不会增加引用计数,但是它能够通过lock()方法来判断它所管理的资源是否被释放。另外很自然地一个问题是:既然weak_ptr不增加资源的引用计数,那么在使用weak_ptr对象的时候,资源被突然释放了怎么办呢?呵呵,答案是你根本不能直接通过weak_ptr来访问资源。那么如何通过weak_ptr来间接访问资源呢?答案是:在需要访问资源的时候weak_ptr为你生成一个shared_ptr,shared_ptr能够保证在shared_ptr没有被释放之前,其所管理的资源是不会被释放的。创建shared_ptr的方法就是lock()方法。

细节:shared_ptr实现了operator bool() const方法来判断一个管理的资源是否被释放


见如下例子

// circleRef.cpp : 定义控制台应用程序的入口点。//#include "stdafx.h"#include <iostream>#include <boost/shared_ptr.hpp>#include <boost/weak_ptr.hpp>using namespace std;class ObjectB;class ObjectA{public:~ObjectA(){std::cout<<"dctor ~ObjectA"<<std::endl;}void setObjectB(boost::shared_ptr<ObjectB> b){m_objB = b;}private:boost::shared_ptr<ObjectB> m_objB;};class ObjectB{public:~ObjectB(){cout<<"dctor ~ObjectB"<<endl;}void setObjectA(boost::shared_ptr<ObjectA> a){m_objA = a;}private:boost::shared_ptr<ObjectA> m_objA;};void test(){boost::shared_ptr<ObjectA> a(new ObjectA);boost::shared_ptr<ObjectB> b(new ObjectB);a->setObjectB(b);b->setObjectA(a);</span>};int _tmain(int argc, _TCHAR* argv[]){test();printf("test ");getchar();return 0;}

程序执行,不会调用各自的析构函数。若将其改为weak_ptr则不会引入多余计数的问题,从而可以保证对象被正确释放,看如下修改后的正确代码:

// circleRef.cpp : 定义控制台应用程序的入口点。//#include "stdafx.h"#include <iostream>#include <boost/shared_ptr.hpp>#include <boost/weak_ptr.hpp>using namespace std;class ObjectB;class ObjectA{public:~ObjectA(){std::cout<<"dctor ~ObjectA"<<std::endl;}void setObjectB(boost::shared_ptr<ObjectB> b){m_objB = b;}private:boost::weak_ptr<ObjectB> m_objB;};class ObjectB{public:~ObjectB(){cout<<"dctor ~ObjectB"<<endl;}void setObjectA(boost::shared_ptr<ObjectA> a){m_objA = a;}private:boost::weak_ptr<ObjectA> m_objA;};void test(){boost::shared_ptr<ObjectA> a(new ObjectA);boost::shared_ptr<ObjectB> b(new ObjectB);a->setObjectB(b);b->setObjectA(a);};int _tmain(int argc, _TCHAR* argv[]){test();printf("test ");getchar();return 0;}


0 0
原创粉丝点击