在android 中可以廣泛看到的template<typename T> class Sp 句柄類實際上是android 為實現垃圾回收機制的智能指針。智能指針是c++ 中的一個概念,因為c++ 本身不具備垃圾回收機制,而且指針也不具備構造函數和析構函數,所以為了實現內存( 動態存儲區) 的安全回收,必須對指針進行一層封裝,而這個封裝就是智能指針,其實說白了,智能指針就是具備指針功能同時提供安全內存回收的一個類。當然,智能指針的功能還不只這些,想了解更多大家可以去研究下~
智能指針有很多實現方式,android 中的sp 句柄類實際上就是google 實現的一種強引用的智能指針。我沒有仔細看android sp 的實現方式,但其基本原理是固定的,現在我們從一個相對簡單的例子來看智能指針的實現:
首先看一個最簡單的對指針的封裝:
class SmartPtr{
public:
SmartPtr(T *p = 0):ptr(p){}
~SmartPtr(){delete ptr ;}
private:
T *ptr ;
};
SmartPtr<int> pointer(new int) ;
*(pointer.ptr) = 10 ;
為了方便使用,我們可以對操作符進行重載,讓智能指針的操作更像是指針:
T &operator*(){return *ptr}
T* operator->(){return ptr}
經過上面的重載,我們就可以像使用真正的指針一樣而不需要去解決 內存泄露問題。
因為智能指針封裝了指針,所以必須為其實現拷貝構造函數和“=”操作符重載。因為如果不提供這兩個函數,當上面的智能指針進行賦值的時候必然會使指針指向同一個區域,一旦析構的話會導致同一個指針被delete 兩次。
在這里,拷貝構造函數的實現是有技巧 的,使用拷貝構造函數創建一個新的只能指針時,并不建立新的對象,而是讓新的智能指針指向同一個對象,實際上就是常說的淺復制。但是這樣的話就會導致多個指針指向同一塊內存區域,當調用析構函數的時候如何來保證同一塊內存區域只會被delete 一次呢,這里實現的方法有很多,最常用的是引數控制。即在智能指針中加入一個計數器,每次增加一個對內存區域的強引用,則計數器加一,當計數器為0 時,這個對象就可以被刪除了, 這個計數器采用動態分配跟指針分開存儲, 因為這個計數器是多個智能指針需要共享的:
class SmartPtr{
public:
SmartPtr(T *p = 0):ptr(p){count = new int(1) ;}// 第一次創建的時候,引數肯定是1
SmartPtr(const SmartPtr & rhs):ptr(rhs.ptr),count(rhs.count){++*rhs.count ;}
T &operator*(){return *ptr}
T* operator->(){return ptr}
SmartPtr &operator=(const SmartPtr & rhs){
if(ptr == rhs.ptr)
return *this ;
if(--*count == 0){
delete ptr ;
delete count ;
}
++*rhs.count ;
count = rhs.count ;
ptr = rhs.ptr ;
}
~SmartPtr(){
if(--*count==0)
delete ptr ;
delete count ;
}
private:
T *ptr ;
int *count ;
};
出了智能指針sp 外,android 里面還出現了wp 這個指針類,實際上他是一個弱引用類型的指針類,弱引用是在.net 以及java 中經常用到的,弱引用是一個對象引用的持有者,使用弱引用后可以維持對對象的引用,但是不會阻止其被垃圾回收。如果一個對象只有弱引用了,那它就成為被垃圾回收的候選對象,就像沒有剩余的引用一樣,而且一旦對象被刪除,所有的弱引用也會被清楚。弱引用適合那些數據 成員特別多,而且重新創建又相對容易的類,也就是俗稱的胖子類,建立弱引用可以引用對象,但也不阻止其被垃圾回收,在內存的使用方面取得一定的平衡。
在android 中wp 類里面的promote 函數實際上就是將一個弱引用升級為一個強引用。不管是sp 還是wp ,實際上都是android 利用現有的c++ 特性來解決內存使用和回收的一種手段。
新聞熱點
疑難解答
圖片精選