Android開發(fā)優(yōu)化之的強(qiáng)引用、軟引用、弱引用的使用

Android開發(fā)優(yōu)化之的強(qiáng)引用、軟引用、弱引用的使用

Anderson大碼渣關(guān)注

2016.12.27 22:42*字?jǐn)?shù) 1951閱讀 4371評論 16喜歡 122

nougat.png

引言

早在JDK1.2,Java就把對象的引用分為四種級別,從而使程序能更加靈活的控制對象的生命周期。這四種級別由高到低依次為:強(qiáng)引用、軟引用、弱引用和虛引用。

但是平時我們的代碼中似乎很少出現(xiàn)這些, 而之前還看到過一份代碼中, 一個Activity中有一個靜態(tài)變量持有對自己的弱引用,來達(dá)到類似的singleTask的效果.

So, 是時候系統(tǒng)的學(xué)習(xí)一下軟引用、弱引用這些,并對我們的代碼進(jìn)行優(yōu)化了.

強(qiáng)引用

String str =newString("xiamin");

強(qiáng)引用是使用最普遍的引用。如果一個對象具有強(qiáng)引用,那垃圾回收器絕不會回收它。當(dāng)內(nèi)存空間不足,Java虛擬機(jī)寧愿拋出OutOfMemoryError錯誤,使程序異常終止,也不會靠隨意回收具有強(qiáng)引用的對象來解決內(nèi)存不足的問題。

強(qiáng)引用特點(diǎn):

強(qiáng)引用可以直接訪問目標(biāo)對象。

強(qiáng)引用所指向的對象在任何時候都不會被系統(tǒng)回收。JVM寧愿拋出OOM異常,也不會回收強(qiáng)引用所指向的對象。

強(qiáng)引用可能導(dǎo)致內(nèi)存泄露。

軟引用 (SoftReference)

簡單用法

MyObject aRef =newMyObject();? ? SoftReference aSoftRef=newSoftReference(aRef);? ? ? ? MyObject anotherRef=(MyObject)aSoftRef.get();

java.lang.ref包中提供了幾個類:SoftReference類、WeakReference類和PhantomReference類,它們分別代表軟引用、弱引用和虛引用。ReferenceQueue類表示引用隊(duì)列,它可以和這三種引用類聯(lián)合使用,以便跟蹤Java虛擬機(jī)回收所引用的對象的活動。

如果一個對象只具有軟引用,那么如果內(nèi)存空間足夠,垃圾回收器就不會回收它;如果內(nèi)存空間不足了,就會回收這些對象的內(nèi)存。只要垃圾回收器沒有回收它,該對象就可以被程序使用。軟引用可用來實(shí)現(xiàn)內(nèi)存敏感的高速緩存。軟引用可以和一個引用隊(duì)列(ReferenceQueue)聯(lián)合使用,如果軟引用所引用的對象被垃圾回收,Java虛擬機(jī)就會把這個軟引用加入到與之關(guān)聯(lián)的引用隊(duì)列中。

ReferenceQueue queue =newReferenceQueue();? ? SoftReference? ref=newSoftReference(aMyObject, queue);

那么當(dāng)這個SoftReference所軟引用的aMyOhject被垃圾收集器回收的同時,ref所強(qiáng)引用的SoftReference對象被列入ReferenceQueue。也就是說,ReferenceQueue中保存的對象是Reference對象,而且是已經(jīng)失去了它所軟引用的對象的Reference對象。另外從ReferenceQueue這個名字也可以看出,它是一個隊(duì)列,當(dāng)我們調(diào)用它的poll()方法的時候,如果這個隊(duì)列中不是空隊(duì)列,那么將返回隊(duì)列前面的那個Reference對象。

在任何時候,我們都可以調(diào)用ReferenceQueue的poll()方法來檢查是否有它所關(guān)心的非強(qiáng)可及對象被回收。如果隊(duì)列為空,將返回一個null,否則該方法返回隊(duì)列中前面的一個Reference對象。利用這個方法,我們可以檢查哪個SoftReference所軟引用的對象已經(jīng)被回收。于是我們可以把這些失去所軟引用的對象的SoftReference對象清除掉。常用的方式為:

SoftReference ref =null;while((ref = (EmployeeRef) q.poll()) !=null) {// 清除ref}

當(dāng)然,我們作為Android開發(fā)中使用,正常是用來處理圖片這種占用內(nèi)存大的類的.

所以,我們應(yīng)該這樣使用.

View view = findViewById(R.id.button);? ? Bitmap bitmap = BitmapFactory.decodeResource(getResources(),R.drawable.ic_launcher);? ? Drawable drawable =newBitmapDrawable(bitmap);? ? SoftReference drawableSoftReference =newSoftReference(drawable);? ? ? ? Drawable bgdrawable = drawableSoftReference.get();if(bgdrawable !=null) {? ? ? ? view.setBackground(bgdrawable);? ? }

這樣的好處是

通過軟引用的get()方法,取得drawable對象實(shí)例的強(qiáng)引用,發(fā)現(xiàn)對象被未回收。在GC在內(nèi)存充足的情況下,不會回收軟引用對象。此時view的背景顯示.

實(shí)際情況中,我們會獲取很多圖片.然后可能給很多個view展示, 這種情況下很容易內(nèi)存吃緊導(dǎo)致oom,

內(nèi)存吃緊,系統(tǒng)開始會GC。這次GC后,drawables.get()不再返回Drawable對象,而是返回null,這時屏幕上背景圖不顯示,說明在系統(tǒng)內(nèi)存緊張的情況下,軟引用被回收。

使用軟引用以后,在OutOfMemory異常發(fā)生之前,這些緩存的圖片資源的內(nèi)存空間可以被釋放掉的,從而避免內(nèi)存達(dá)到上限,避免Crash發(fā)生。

需要注意的是,在垃圾回收器對這個Java對象回收前,SoftReference類所提供的get方法會返回Java對象的強(qiáng)引用,一旦垃圾線程回收該Java對象之后,get方法將返回null。所以在獲取軟引用對象的代碼中,一定要判斷是否為null,以免出現(xiàn)NullPointerException異常導(dǎo)致應(yīng)用崩潰。

弱引用 (WeakReference)

用法

WeakReference sr =newWeakReference(newUser());

如果一個對象只具有弱引用,那么在垃圾回收器線程掃描的過程中,一旦發(fā)現(xiàn)了只具有弱引用的對象,不管當(dāng)前內(nèi)存空間足夠與否,都會回收它的內(nèi)存。不過,由于垃圾回收器是一個優(yōu)先級很低的線程,因此不一定會很快發(fā)現(xiàn)那些只具有弱引用的對象。弱引用也可以和一個引用隊(duì)列(ReferenceQueue)聯(lián)合使用,如果弱引用所引用的對象被垃圾回收,Java虛擬機(jī)就會把這個弱引用加入到與之關(guān)聯(lián)的引用隊(duì)列中。

弱引用與軟引用的根本區(qū)別在于:只具有弱引用的對象擁有更短暫的生命周期,可能隨時被回收。而只具有軟引用的對象只有當(dāng)內(nèi)存不夠的時候才被回收,在內(nèi)存足夠的時候,通常不被回收。

使用場景,handler的使用防止內(nèi)存泄露

importandroid.os.Bundle;importandroid.os.Handler;importandroid.os.Message;importandroid.support.v7.app.AppCompatActivity;importjava.lang.ref.WeakReference;publicclassMainActivityextendsAppCompatActivity{privateHandler handler? ;@OverrideprotectedvoidonCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);? ? ? ? setContentView(R.layout.activity_main);? ? ? ? handler =newMyHandler(this) ;newThread(newRunnable() {@Overridepublicvoidrun(){? ? ? ? ? ? ? handler.sendEmptyMessage(0) ;? ? ? ? ? ? }? ? ? ? }).start() ;? ? }privatestaticclassMyHandlerextendsHandler{? ? ? ? WeakReference weakReference ;publicMyHandler(MainActivity activity ){? ? ? ? ? ? weakReference? =newWeakReference( activity) ;? ? ? ? }@OverridepublicvoidhandleMessage(Message msg){super.handleMessage(msg);if( weakReference.get() !=null){// update android ui}? ? ? ? }? ? } }

在Android應(yīng)用的開發(fā)中,為了防止內(nèi)存溢出,在處理一些占用內(nèi)存大而且聲明周期較長的對象時候,可以盡量應(yīng)用軟引用和弱引用技術(shù)。

軟引用,弱引用都非常適合來保存那些可有可無的緩存數(shù)據(jù)。如果這樣做,當(dāng)系統(tǒng)內(nèi)存不足時,這些緩存數(shù)據(jù)會被回收,不會導(dǎo)致內(nèi)存溢出。而當(dāng)內(nèi)存資源充足時,這些緩存數(shù)據(jù)又可以存在相當(dāng)長的時間。

到底什么時候使用軟引用,什么時候使用弱引用呢?

個人認(rèn)為,如果只是想避免OutOfMemory異常的發(fā)生,則可以使用軟引用。如果對于應(yīng)用的性能更在意,想盡快回收一些占用內(nèi)存比較大的對象,則可以使用弱引用。

還有就是可以根據(jù)對象是否經(jīng)常使用來判斷。如果該對象可能會經(jīng)常使用的,就盡量用軟引用。如果該對象不被使用的可能性更大些,就可以用弱引用。

另外,和弱引用功能類似的是WeakHashMap。WeakHashMap對于一個給定的鍵,其映射的存在并不阻止垃圾回收器對該鍵的回收,回收以后,其條目從映射中有效地移除。WeakHashMap使用ReferenceQueue實(shí)現(xiàn)的這種機(jī)制。

謝謝大家閱讀,如有幫助,來個喜歡或者關(guān)注吧!

本文作者:Anderson/Jerey_Jobs

簡書地址? :Anderson大碼渣

CSDN地址? :Jerey_Jobs的專欄

github地址 :Jerey_Jobs

? ? ? ? ? ? ? 上一篇


目錄

? ? ? ? ? ? ? 下一篇


小禮物走一走,來簡書關(guān)注我

贊賞支持

andriod學(xué)習(xí)

? ? ? ? ? ? 著作權(quán)歸作者所有


舉報文章

? ? ? ? ? ? ? andriod學(xué)習(xí)


5.2萬字 ·3.9萬閱讀·113人關(guān)注



關(guān)注

喜歡

122

? ? ? ?

? ? ? ?

? ? ? ? ? 更多分享

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容