什么是內存泄露丁侄?
Java使用有向圖機制,通過GC自動檢查內存中的對象(什么時候檢查由虛擬機決定),如果GC發(fā)現一個或一組對象為不可到達狀態(tài)惑芭,則將該對象從內存中回收。也就是說继找,一個對象不被任何引用所指向遂跟,則該對象會在被GC發(fā)現的時候被回收;另外婴渡,如果一組對象中只包含互相的引用幻锁,而沒有來自它們外部的引用(例如有兩個對象A和B互相持有引用,但沒有任何外部對象持有指向A或B的引用)边臼,這仍然屬于不可到達哄尔,同樣會被GC回收。
Android中使用Handler造成內存泄露的原因
Handler mHandler =newHandler(){
? ? ? ? ?@Override
? ? ? ? ? publicvoidhandleMessage(Message msg){? ? ? ??
? ? ? ? ? ? ? ?mImageView.setImageBitmap(mBitmap);? ?
? ? ? ? ? }
}
上面是一段簡單的Handler的使用柠并。當使用內部類(包括匿名類)來創(chuàng)建Handler的時候岭接,Handler對象會隱式地持有一個外部類對象(通常是一個Activity)的引用(不然你怎么可能通過Handler來操作Activity中的View富拗?)。而Handler通常會伴隨著一個耗時的后臺線程(例如從網絡拉取圖片)一起出現鸣戴,這個后臺線程在任務執(zhí)行完畢(例如圖片下載完畢)之后啃沪,通過消息機制通知Handler,然后Handler把圖片更新到界面窄锅。然而创千,如果用戶在網絡請求過程中關閉了Activity,正常情況下入偷,Activity不再被使用追驴,它就有可能在GC檢查時被回收掉,但由于這時線程尚未執(zhí)行完疏之,而該線程持有Handler的引用(不然它怎么發(fā)消息給Handler殿雪?),這個Handler又持有Activity的引用体捏,就導致該Activity無法被回收(即內存泄露)冠摄,直到網絡請求結束(例如圖片下載完畢)。另外几缭,如果你執(zhí)行了Handler的postDelayed()方法河泳,該方法會將你的Handler裝入一個Message,并把這條Message推到MessageQueue中年栓,那么在你設定的delay到達之前拆挥,會有一條MessageQueue -> Message -> Handler -> Activity的鏈,導致你的Activity被持有引用而無法被回收某抓。
內存泄露的危害
只有一個纸兔,那就是虛擬機占用內存過高,導致OOM(內存溢出)否副,程序出錯汉矿。對于Android應用來說,就是你的用戶打開一個Activity备禀,使用完之后關閉它洲拇,內存泄露;又打開曲尸,又關閉赋续,又泄露;幾次之后另患,程序占用內存超過系統(tǒng)限制纽乱,FC。
使用Handler導致內存泄露的解決方法
方法一:通過程序邏輯來進行保護昆箕。
1.在關閉Activity的時候停掉你的后臺線程鸦列。線程停掉了租冠,就相當于切斷了Handler和外部連接的線,Activity自然會在合適的時候被回收薯嗤。
2.如果你的Handler是被delay的Message持有了引用肺稀,那么使用相應的Handler的removeCallbacks()方法,把消息對象從消息隊列移除就行了应民。
方法二:將Handler聲明為靜態(tài)類。
靜態(tài)類不持有外部類的對象夕吻,所以你的Activity可以隨意被回收诲锹。代碼如下:
static class MyHandler extends Handler{
? ? ?@Override
? ? ?public void handleMessage(Messagemsg) {??
? ? ? ? ? ? ? mImageView.setImageBitmap(mBitmap);
? ? }
}
但其實沒這么簡單。使用了以上代碼之后涉馅,你會發(fā)現归园,由于Handler不再持有外部類對象的引用,導致程序不允許你在Handler中操作Activity中的對象了稚矿。所以你需要在Handler中增加一個對Activity的弱引用(WeakReference):
static class MyHandler extends Handler{
? ? ? ? ?WeakReference mActivityReference;
? ? ? ? ?MyHandler(Activity activity) {? ? ? ?
? ? ? ? ? ? ? mActivityReference=new WeakReference(activity);?
? ? ? ? ?}
? ? ? ? @Override
? ? ? ? ?public void handleMessage(Messagemsg) {
? ? ? ? ? ? ?final Activity activity = mActivityReference.get();
? ? ? ? ? ? ?if(activity !=null) { mImageView.setImageBitmap(mBitmap); ?}??
? ? ? ? }
}
將代碼改為以上形式之后庸诱,就算完成了。
延伸:什么是WeakReference晤揣?
WeakReference弱引用桥爽,與強引用(即我們常說的引用)相對,它的特點是昧识,GC在回收時會忽略掉弱引用钠四,即就算有弱引用指向某對象,但只要該對象沒有被強引用指向(實際上多數時候還要求沒有軟引用跪楞,但此處軟引用的概念可以忽略)缀去,該對象就會在被GC檢查到時回收掉。對于上面的代碼甸祭,用戶在關閉Activity之后缕碎,就算后臺線程還沒結束,但由于僅有一條來自Handler的弱引用指向Activity池户,所以GC仍然會在檢查的時候把Activity回收掉咏雌。這樣,內存泄露的問題就不會出現了煞檩。