shared_ptr 配合 weak_ptr 解决环形内存问题
来源:互联网 发布:snow软件下载 编辑:程序博客网 时间:2024/06/06 16:10
<1>
1,
#ifndef __JNTest__JNShared_Ptr__#define __JNTest__JNShared_Ptr__#include <iostream>using namespace std;#include<memory.h>class Woman;class Man{public: void setWife(shared_ptr<Woman> woman){ _wife = woman; } void doSomething(){ if(_wife.lock()){ } } ~Man(){ log("kill man"); }private: weak_ptr<Woman> _wife;};//class Woman{public: void setHusband(shared_ptr<Man> man){ _husband = man; } ~Woman(){ log("kill woman"); }private: shared_ptr<Man> _husband;};#endif /* defined(__JNTest__JNShared_Ptr__) */2,
void HelloWorld::Main(float dt){ shared_ptr<Man> m(new Man()); shared_ptr<Woman> w(new Woman()); if(m && w){ m->setWife(w); w->setHusband(m); } log("<<<m:%d", (int)m.use_count()); log("<<<w:%d", (int)w.use_count());}/* cocos2d: <<<m:2 cocos2d: <<<w:1 cocos2d: kill woman cocos2d: kill man*/
<2>
1,
#ifndef __JNTest__JNShared_Ptr__#define __JNTest__JNShared_Ptr__#include <iostream>using namespace std;#include<memory.h>class Woman;class Man{public: void setWife(shared_ptr<Woman> woman){ _wife = woman; } void doSomething(){// if(_wife.lock()){// // } } ~Man(){ log("kill man"); }private: shared_ptr<Woman> _wife;};//class Woman{public: void setHusband(shared_ptr<Man> man){ log("<<<1:%d", (int)man.use_count()); _husband = man; log("<<<2:%d", (int)man.use_count()); log("<<<3:%d", (int)_husband.use_count()); } ~Woman(){ log("kill woman"); }private: shared_ptr<Man> _husband;};#endif /* defined(__JNTest__JNShared_Ptr__) */2,
void HelloWorld::Main(float dt){ shared_ptr<Man> m(new Man()); shared_ptr<Woman> w(new Woman()); if(m && w){ m->setWife(w); w->setHusband(m); } log("<<<m:%d", (int)m.use_count()); log("<<<w:%d", (int)w.use_count());}/* cocos2d: <<<1:2 cocos2d: <<<2:3 cocos2d: <<<3:3 cocos2d: <<<m:2 cocos2d: <<<w:2*/
总结:
<1>
可见,第1种情况出现了内存泄漏,注意局部的shared_ptr在使用后就释放了,所以m最后依然是:2。
在std::shared_ptr被引入之前,C++标准库中实现的用于管理资源的智能指针只有std::auto_ptr一个而已。std::auto_ptr的作用非常有限,因为它存在被管理资源的所有权转移问题。这导致多个std::auto_ptr类型的局部变量不能共享同一个资源,这个问题是非常严重的哦。因为,我个人觉得,智能指针内存管理要解决的根本问题是:一个堆对象(或则资源,比如文件句柄)在被多个对象引用的情况下,何时释放资源的问题。何时释放很简单,就是在最后一个引用它的对象被释放的时候释放它。关键的问题在于无法确定哪个引用它的对象是被最后释放的。std::shared_ptr确定最后一个引用它的对象何时被释放的基本想法是:对被管理的资源进行引用计数,当一个shared_ptr对象要共享这个资源的时候,该资源的引用计数加1,当这个对象生命期结束的时候,再把该引用技术减少1。这样当最后一个引用它的对象被释放的时候,资源的引用计数减少到0,此时释放该资源。
在Man类内部会引用一个Woman,Woman类内部也引用一个Man。当一个man和一个woman是夫妻的时候,他们直接就存在了相互引用问题。man内部有个用于管理wife生命期的shared_ptr变量,也就是说wife必定是在husband去世之后才能去世。同样的,woman内部也有一个管理husband生命期的shared_ptr变量,也就是说husband必须在wife去世之后才能去世。这就是循环引用存在的问题:husband的生命期由wife的生命期决定,wife的生命期由husband的生命期决定,最后两人都死不掉,违反了自然规律,导致了内存泄漏。
解决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方法来判断一个管理的资源是否被释放。
<2>
1,肯定不会触发析构函数啊。既然new B的一用计数是1,当然这个new来的对象就不会释放。。。所以shared_ptr<A> _a它的生命周期就没有结束啊
2,这样的话。。shared_ptr<B> _b这里改成weak_ptr<B> _b这样的话,B的引用计数是1,A的引用计数是2。。这样的话,当shared_ptr<B> b;没了时,b的引用计数减去1则为0,导致new B对象呗析构,然后接着就可以析构里面的shared_ptr<A> _a.从而new A的饮用计数减去1,为1,然后犹豫shared_ptr<A > a的生命期结束,导致new A饮用计数又减去1.。从而new A计数也为0了,完美释放内存
3,注意shared_ptr生命周期结束了,只是减少引用计数,当饮用计数为0了,就去看里面的东西,不然不用看对象里面的数据成员,因为饮用计数又不为0,肯定不会析构啊!
- shared_ptr 配合 weak_ptr 解决环形内存问题
- 用weak_ptr解决shared_ptr的环形引用问题
- weak_ptr解决shared_ptr环状引用所引起的内存泄漏
- weak_ptr解决shared_ptr环状引用所引起的内存泄漏
- weak_ptr解决shared_ptr环状引用所引起的内存泄漏
- weak_ptr解决shared_ptr环状引用所引起的内存泄漏
- C++智能指针(三):weak_ptr--解决shared_ptr循环引用问题
- 弱引用weak_ptr 解决shared_ptr的循环引用
- 弱引用weak_ptr解决shared_ptr的循环引用
- enable_shared_from_this、weak_ptr、shared_ptr
- auto_ptr,scoped_ptr,shared_ptr,weak_ptr
- weak_ptr,shared_ptr,scoped_ptr
- auto_ptr,shared_ptr,weak_ptr,scoped_ptr
- 智能指针--shared_ptr&&weak_ptr
- shared_ptr or weak_ptr
- weak_ptr与shared_ptr
- shared_ptr,weak_ptr源码解析
- shared_ptr,weak_ptr,unique_ptr
- 无线个人通信(WPAN)-蓝牙
- spring和hibernate整合时不能自动创建表的问题
- It's never too late
- UVa11292
- 【Visual Basic】纯代码不拖控件,利用动态生成控件的方式完成一个简单的四则运算计算器
- shared_ptr 配合 weak_ptr 解决环形内存问题
- apk和odex的介绍和合并
- OpenCV(2.4.11)的安装与配置
- poj2796Feel Good 单调栈
- 学生信息
- 生成n个1
- android 自定义view,绘制与onTouchEvent事件(一)
- 无线网络(WLAN)- Wi-Fi
- 摄像头的工作原理