`

Android中的sp和wp指针

 
阅读更多

经常会在android的framework代码中发现sp<xxx>和wp<xxx>这样的指针,平时看的时候都把他当成一个普通的指针封装过掉了,这几天终于忍不住了,想深入了解一下。

 

相关的代码:

frameworks/base/include/utils/RefBase.h

frameworks/base/libs/utils/RefBase.cpp

 

sp和wp都是一个模板类,看一下sp类的定义:

 

  1. template <typename T>  
  2. class sp  
  3. {  
  4. public:  
  5.     typedef typename RefBase::weakref_type weakref_type;  
  6.     inline sp() : m_ptr(0) { }  
  7.     sp(T* other);  
  8.     sp(const sp<T>& other);  
  9.     ~sp();  
  10.     ......  
  11. private:      
  12.     // Optimization for wp::promote().  
  13.     sp(T* p, weakref_type* refs);  
  14.       
  15.     T*              m_ptr;  
  16. };  

 

 

可以看到他确实封转了一个原生指针T* m_ptr. 再看一下其中一个构造函数和析构函数:

 

 

[c-sharp] view plaincopy
  1. template<typename T>  
  2. sp<T>::sp(T* other)  
  3.     : m_ptr(other)  
  4. {  
  5.     if (other) other->incStrong(this);  
  6. }  
  7.   
  8. template<typename T>  
  9. sp<T>::~sp()  
  10. {  
  11.     if (m_ptr) m_ptr->decStrong(this);  
  12. }  

 

 

咋一看好奇怪,因为在构造函数中调用了incStrong(),在析构函数中调用的decStrong(),显然是管理引用计数的函数,但是sp类的中并没有定义这两个函数,这两个函数是在RefBase类中定义的,由此可以得出结论:

要想使用sp<T>或者wp<T>, T必需要继承RefBase类才行

 

RefBase的静态关系如下:

 

 

 

 其中weakref_type是RefBase的内嵌类,weakref_impl则是weakref_type的子类,RefBase的大部分工作都是交由weakref_impl类来完成,通过RefBase的成员变量weakref_impl* const mRefs。查看其中一个sp的构造函数:

 

 

[c-sharp] view plaincopy
  1. template<typename T>  
  2. sp<T>::sp(T* other)  
  3.     : m_ptr(other)  
  4. {  
  5.     if (other) other->incStrong(this);  
  6. }  

 

 

建立sp<xxx>的动态关系如下:

sp<T>

  --> RefBase : incStrong()

  -->weakref_impl : addStrongRef()

  -->android_atomic_inc(&refs->mStrong)

 

可见当一个普通指针变成一个sp指针后,将会由RefBase类维护该指针的引用计数,当引用为零时则自动释放该指针指向的内存:

 

[c-sharp] view plaincopy
  1. void RefBase::decStrong(const void* id) const  
  2. {  
  3.     weakref_impl* const refs = mRefs;  
  4.     refs->removeStrongRef(id);  
  5.     const int32_t c = android_atomic_dec(&refs->mStrong);  
  6.     if (c == 1) {  
  7.         const_cast<RefBase*>(this)->onLastStrongRef(id);  
  8.         if ((refs->mFlags&OBJECT_LIFETIME_WEAK) != OBJECT_LIFETIME_WEAK) {  
  9.             delete this;    //引用为0,销毁  
  10.         }  
  11.     }  
  12.     refs->removeWeakRef(id);  
  13.     refs->decWeak(id);  
  14. }  

 

 

 wp<xxx>是怎么一回事?

wp其实是弱指针的意思,wp<T>类型不能直接对类型T进行操作,要想对T进行某种操作,必需把wp升级为sp指针,使用promote()来实现升级:

        wp<T> weakp= new T();

        sp<T> t = weakp.promote();

wp可能会在弱引用计数不为0的情况下被销毁,执行如下代码:

 

 

[c-sharp] view plaincopy
  1. class WPTest : public RefBase {  
  2. public:  
  3.     WPTest(){  
  4.         LOGD("WPTest constructor");  
  5.     }  
  6.     virtual ~WPTest() {  
  7.         LOGD("WPTest destructor");  
  8.     }  
  9.   
  10.     virtual void onFirstRef() {  
  11.         LOGD("first weak ptr ref callback");  
  12.     }  
  13.   
  14.     virtual void onLastStrongRef(const void* id) {  
  15.         LOGD("last strong ptr ref callback");  
  16.     }  
  17.   
  18.     virtual void onLastWeakRef(const void* id) {  
  19.         LOGD("last weak ptr ref callback");  
  20.     }  
  21. };  
  22.   
  23.   
  24. int main()  
  25. {  
  26.     WPTest *T = new WPTest();  
  27.     {  
  28.         wp<WPTest> weakp(T);  
  29.   
  30.         {  
  31.             LOGD("promote to strong ptr.../n");  
  32.   
  33.             sp<WPTest> strongp = weakp.promote();  
  34.   
  35.             LOGD("strong ptr's lifetime is just about to finish .../n");  
  36.         }  
  37.   
  38.         LOGD("weak ptr's lifetime is just about to finish .../n");  
  39.     }  
  40.   
  41.     LOGD("weak ptr is out of scope./n");  
  42.   
  43.     return 0;  
  44. }  

 

程序打印的结果是:

D/sp-wp-sample(  225): WPTest constructor
D/sp-wp-sample(  225): promote to strong ptr...
D/sp-wp-sample(  225): first weak ptr ref callback
D/sp-wp-sample(  225): strong ptr's lifetime is just about to finish ...
D/sp-wp-sample(  225): last strong ptr ref callback
D/sp-wp-sample(  225): WPTest destructor
D/sp-wp-sample(  225): weak ptr's lifetime is just about to finish ...
D/sp-wp-sample(  225): weak ptr is out of scope.

由此可见虽然wp<WPTest >的生命周期还没有结束,但是因为升级为sp<WPTest >后,sp<WPTest >的强引用计数为0,导致WPTest 被销毁,当强引用为0而弱引用不为0时,WPTest 销毁时,基类RefBase的mRefs指向的weakref_impl类并没有释放,从而保证了弱引用可以继续起作用,这点可以从RefBase的析构函数中看出来:

 

[c-sharp] view plaincopy
  1. RefBase::~RefBase()  
  2. {  
  3. //    LOGV("Destroying RefBase %p (refs %p)/n", this, mRefs);  
  4.     if (mRefs->mWeak == 0) {  
  5. //        LOGV("Freeing refs %p of old RefBase %p/n", mRefs, this);  
  6.         delete mRefs;  
  7.     }  
  8. }  

 

 

不过也可以改变这一行为,我们修改一下WPTest的构造函数:

 

[c-sharp] view plaincopy
  1. WPTest(){  
  2.         LOGD("WPTest constructor");  
  3.         extendObjectLifetime(OBJECT_LIFETIME_WEAK);  
  4.     }  

 

 

这时的打印结果是:

D/sp-wp-sample(  217): WPTest constructor
D/sp-wp-sample(  217): promote to strong ptr...
D/sp-wp-sample(  217): first weak ptr ref callbac
D/sp-wp-sample(  217): strong ptr's lifetime is j
D/sp-wp-sample(  217): last strong ptr ref callba
D/sp-wp-sample(  217): weak ptr's lifetime is j
D/sp-wp-sample(  217): last weak ptr ref callback
D/sp-wp-sample(  217): WPTest destructor
D/sp-wp-sample(  217): weak ptr is out of scope.

 

可以看出现在只有当强引用和弱引用的计数都为0时,WPTest对象才会被销毁。


分享到:
评论

相关推荐

    anroid智能指针(wp,sp)学习总结

    智能指针sp和wp在android c++源码中使用非常频繁,例如IBinder机制,但是它比c++中普通的智能指针要复杂很多,相信不少android学习者如果c++基础不是很扎实的,看起来会比较吃力和枯燥。本人在android 4.2.2源码基础...

    理解 Android sp wp 指针

    在Android中,`sp`和`wp`是C++智能指针的缩写,`sp`代表`std::shared_ptr`,`wp`代表`std::weak_ptr`,它们是C++11标准库中的智能指针类型。`std::shared_ptr`是强引用,而`std::weak_ptr`是弱引用。 3. **std::...

    Android_WP_SP浅析

    在Android的Native层开发中,为了管理和防止内存泄漏,Google引入了一套基于C++的智能指针系统,主要包含两个关键类型:sp(Strong Pointer)和wp(Weak Pointer)。它们都是基于类RefBase的引用计数机制来实现的,...

    android智能指针详解[收集].pdf

    在Android系统中,智能指针是用来管理对象生命周期的重要工具,主要分为`sp`(strong pointer)和`wp`(weak pointer)两种类型。它们都是基于C++的引用计数机制,用于自动管理内存,防止内存泄漏。在Android框架层,...

    安卓智能指针测试 spwp.tar

    包含RefBase.h,RefBase.cpp,WeakPointer.h,StrongPointer.h,LightRefBase,模拟安卓原子操作的android_atomic.cpp,android_atomic.h,所有文件组成了安卓智能指针sp,wp,LightRefBase

    Android垃圾回收实质内容解析实用教案.ppt

    这份“Android垃圾回收实质内容解析实用教案”主要探讨了Android中智能指针的使用,特别是`sp`(strong pointer)和`wp`(weak pointer)如何协同工作以支持垃圾回收机制。 `sp`是Android中用于管理对象生命周期的...

    Android垃圾回收实质内容解析学习教案.pptx

    本文将深入探讨Android垃圾回收的实质内容,特别是通过智能指针(如sp和wp)来理解和实现这一机制。 首先,所有Android的类都会隐式地继承自`refbase`类。`refbase`提供了垃圾回收所需的功能,使得每个对象在声明后...

    Android垃圾回收实质内容解析PPT教学课件.pptx

    本篇将深入解析Android垃圾回收的实质内容,主要关注智能指针的使用,包括`sp`和`wp`类型,它们在Android内存管理中的作用。 首先,Android的对象几乎都会继承`refbase`类,因为`refbase`提供了垃圾回收所需的功能...

    Mstar理论及实践篇参考.pdf

    【Mstar理论及实践篇参考】文档主要涵盖了Mstar开发项目中的智能指针使用,特别是Android系统中的强指针(sp)和弱指针(wp)。智能指针是C++中解决对象生命周期管理的一种机制,避免了传统手动内存管理可能导致的内存...

    Android垃圾回收实质内容解析PPT.ppt

    Android的垃圾回收机制主要是针对Java层的,但对于使用C++编写的部分,Android也提供了一种类似智能指针的机制来实现垃圾回收,这就是PPT中提到的`sp`(Strong Pointer)和`wp`(Weak Pointer)。这两个类是Android...

    Android框架层学习指导

    TypeHelpers中定义了sp和wp模板,它们是智能指针,用于管理对象生命周期,尤其是在内存敏感的环境中。 - `sp&lt;T&gt;`是一个强引用智能指针,通常用于表示对象的所有权。 - `wp&lt;T&gt;`是一个弱引用智能指针,不保证对象的...

    Mstar理论及实践篇

    Android系统源码中使用了特有的智能指针类型sp(强指针)和wp(弱指针),它们都依赖于一个共同的基类RefBase,实现了引用计数的管理机制。 强指针sp类似于其他语言中的智能指针,它通过引用计数记录有多少使用者在...

    Mstar安卓理论及实践篇

    在Android源码中,智能指针以`sp&lt;xxx&gt;`(强指针)和`wp&lt;xxx&gt;`(弱指针)的形式出现,这两个模板类用于管理对象的生命周期,确保在对象不再被需要时能够自动释放。 在智能指针的两种类型中,强指针具有控制对象生命...

    Android垃圾回收实质内容解析PPT学习教案.pptx

    `sp`和`wp`两种智能指针类型帮助开发者更方便地管理内存,防止未被引用的对象占用内存。`sp`负责保持对象的生命周期,而`wp`则允许对对象的非拥有性访问,两者协同工作,确保了Android系统的内存管理效率和程序的...

    智能指针与弱引用详解

    在Android中,`wp`和`Sp`可以配合使用。如果一个对象被多个`Sp`引用,同时也被一个或多个`wp`引用,当所有`Sp`都被销毁后,`wp`仍然可以检测到对象是否已经被删除(通过检查弱引用是否有效)。如果对象依然存在,`wp...

    [文档]深入理解Andorid重难点.ppt

    - **RefBase、sp和wp**:RefBase是Android中用于引用计数管理的基础类,sp和wp是智能指针,分别代表强引用和弱引用,用于管理对象生命周期,防止内存泄漏。 **Binder重难点分析** - **Binder时空穿越魔术揭秘**:...

    Android系统进程间通信Binder机制在低层的C++接口源代码文件分析[归类].pdf

    `sp&lt;T&gt;`和`wp&lt;T&gt;`是两种模板智能指针,分别用于强引用和弱引用。`sp&lt;T&gt;`用于持有对象并自动管理其生命周期,而`wp&lt;T&gt;`则允许在对象可能已经被删除的情况下保持引用。 `IBinder`是Binder接口的核心抽象类,它继承自`...

Global site tag (gtag.js) - Google Analytics