智能指針與弱引用詳解
在android 中可以廣泛看到的template<typename T> class Sp 句柄類實(shí)際上是android 為實(shí)現(xiàn)垃圾回收機(jī)制的智能指針。智能指針是c++ 中的一個概念,因為c++ 本身不具備垃圾回收機(jī)制,而且指針也不具備構(gòu)造函數(shù)和析構(gòu)函數(shù),所以為了實(shí)現(xiàn)內(nèi)存( 動態(tài)存儲區(qū)) 的安全回收,必須對指針進(jìn)行一層封裝,而這個封裝就是智能指針,其實(shí)說白了,智能指針就是具備指針功能同時提供安全內(nèi)存回收的一個類。當(dāng)然,智能指針的功能還不只這些,想了解更多大家可以去研究下~
智能指針有很多實(shí)現(xiàn)方式,android 中的sp 句柄類實(shí)際上就是google 實(shí)現(xiàn)的一種強(qiáng)引用的智能指針。我沒有仔細(xì)看android sp 的實(shí)現(xiàn)方式,但其基本原理是固定的,現(xiàn)在我們從一個相對簡單的例子來看智能指針的實(shí)現(xiàn):
首先看一個最簡單的對指針的封裝:
Template <typename T>
class SmartPtr{
public:
SmartPtr(T *p = 0):ptr(p){}
~SmartPtr(){delete ptr ;}
private:
T *ptr ;
};
通過上面的封裝,我們就可以用下面的方式來使用SmartPtr 而不需要擔(dān)心內(nèi)存泄露的問題:
SmartPtr<int> pointer(new int) ;
*(pointer.ptr) = 10 ;
為了方便使用,我們可以對操作符進(jìn)行重載,讓智能指針的操作更像是指針:
T &operator*(){return *ptr}
T* operator->(){return ptr}
經(jīng)過上面的重載,我們就可以像使用真正的指針一樣而不需要去解決 內(nèi)存泄露問題。
因為智能指針封裝了指針,所以必須為其實(shí)現(xiàn)拷貝構(gòu)造函數(shù)和“=”操作符重載。因為如果不提供這兩個函數(shù),當(dāng)上面的智能指針進(jìn)行賦值的時候必然會使指針指向同一個區(qū)域,一旦析構(gòu)的話會導(dǎo)致同一個指針被delete 兩次。
在這里,拷貝構(gòu)造函數(shù)的實(shí)現(xiàn)是有技巧 的,使用拷貝構(gòu)造函數(shù)創(chuàng)建一個新的只能指針時,并不建立新的對象,而是讓新的智能指針指向同一個對象,實(shí)際上就是常說的淺復(fù)制。但是這樣的話就會導(dǎo)致多個指針指向同一塊內(nèi)存區(qū)域,當(dāng)調(diào)用析構(gòu)函數(shù)的時候如何來保證同一塊內(nèi)存區(qū)域只會被delete 一次呢,這里實(shí)現(xiàn)的方法有很多,最常用的是引數(shù)控制。即在智能指針中加入一個計數(shù)器,每次增加一個對內(nèi)存區(qū)域的強(qiáng)引用,則計數(shù)器加一,當(dāng)計數(shù)器為0 時,這個對象就可以被刪除了, 這個計數(shù)器采用動態(tài)分配跟指針分開存儲, 因為這個計數(shù)器是多個智能指針需要共享的:
Template <typename T>
class SmartPtr{
public:
SmartPtr(T *p = 0):ptr(p){count = new int(1) ;}// 第一次創(chuàng)建的時候,引數(shù)肯定是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 ;
};
這樣,一個智能指針就基本成形了,當(dāng)然這只是最簡單的智能指針,商業(yè)庫提供的智能指針都提供非常強(qiáng)大的功能,如果能仔細(xì)研究透了android 在這方面的實(shí)現(xiàn),應(yīng)該在c++ 內(nèi)存控制方面很有長進(jìn)~~暫時還沒有時間,還要順著camera 往下看,有牛人懂的話email 多指教哈~~
出了智能指針sp 外,android 里面還出現(xiàn)了wp 這個指針類,實(shí)際上他是一個弱引用類型的指針類,弱引用是在.net 以及java 中經(jīng)常用到的,弱引用是一個對象引用的持有者,使用弱引用后可以維持對對象的引用,但是不會阻止其被垃圾回收。如果一個對象只有弱引用了,那它就成為被垃圾回收的候選對象,就像沒有剩余的引用一樣,而且一旦對象被刪除,所有的弱引用也會被清楚。弱引用適合那些數(shù)據(jù) 成員特別多,而且重新創(chuàng)建又相對容易的類,也就是俗稱的胖子類,建立弱引用可以引用對象,但也不阻止其被垃圾回收,在內(nèi)存的使用方面取得一定的平衡。
在android 中wp 類里面的promote 函數(shù)實(shí)際上就是將一個弱引用升級為一個強(qiáng)引用。不管是sp 還是wp ,實(shí)際上都是android 利用現(xiàn)有的c++ 特性來解決內(nèi)存使用和回收的一種手段。
相關(guān)文章
詳解Android中的MVP架構(gòu)分解和實(shí)現(xiàn)
本篇文章主要介紹了詳解Android中的MVP架構(gòu)分解和實(shí)現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-02-02Android開發(fā)之AlertDialog實(shí)現(xiàn)彈出對話框
這篇文章主要為大家詳細(xì)介紹了Android開發(fā)之AlertDialog實(shí)現(xiàn)彈出對話框,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-09-09Android開發(fā)之App widget用法實(shí)例分析
這篇文章主要介紹了Android開發(fā)之App widget用法,結(jié)合實(shí)例形式詳細(xì)分析了Android開發(fā)中使用App widget組件的具體步驟與相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2016-06-06Android獲取移動網(wǎng)絡(luò)信號強(qiáng)度的方法
這篇文章主要介紹了Android獲取移動網(wǎng)絡(luò)信號強(qiáng)度的方法,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-04-04Android實(shí)現(xiàn)類似于PC中的右鍵彈出菜單效果
這篇文章主要介紹了Android實(shí)現(xiàn)類似于PC中的右鍵彈出菜單效果,需要的朋友可以參考下2015-12-12android handler.post和handler.sendMessage的區(qū)別和聯(lián)系
handler.post和handler.sendMessage本質(zhì)上是沒有區(qū)別的,都是發(fā)送一個消息到消息隊列中,而且消息隊列和handler都是依賴于同一個線程的。接下來通過本文給大家分享android handler.post和handler.sendMessage的區(qū)別和聯(lián)系,一起看看吧2017-08-08Android中給fragment寫入?yún)?shù)的輕量開發(fā)包FragmentArgs簡介
這篇文章主要介紹了Android中給fragment寫入?yún)?shù)的輕量開發(fā)包FragmentArgs簡介,需要的朋友可以參考下2014-10-10