WeakReference 在android中的應用 弱引用防止記憶體洩漏
首先我們來看一段程式碼
public classAutoActivityextendsActivity{
Handler handler = new Handler(){
publicvoidhandleMessage(android.os.Message msg){
};
};
@Override
protectedvoidonCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_auto);
}
}
上面這段低嗎沒有什麼問題,但是在handler物件建立的時候卻會報警告:This Handler class should be static or leaks might occur意思是:Handler
類應該為static型別,否則可能會造成記憶體洩漏。為什麼會造成這種情況呢?這種情況就是由於android的特殊機制造成的:當一個android主執行緒被創
建的時候,同時會有一個Looper物件被建立,而這個Looper物件會實現一個MessageQueue(訊息佇列),當我們建立一個handler物件時,而handler的
作用就是放入和取出訊息從這個訊息佇列中,每當我們通過handler將一個msg放入訊息佇列時,這個msg就會持有一個handler物件的引用。因此當
Activity被結束後,這個msg在被取出來之前,這msg會繼續存活,但是這個msg持有handler的引用,而handler在Activity中建立,會持有Activity的引用,
因而當Activity結束後,Activity物件並不能夠被gc回收,因而出現記憶體洩漏。
這個根本原因就是:Activity在被結束之後,MessageQueue並不會隨之被結束,如果這個訊息佇列中存在msg,則導致持有handler的引用,但是又
由於Activity被結束了,msg無法被處理,從而導致永久持有handler物件,handler永久持有Activity物件,於是發生記憶體洩漏。但是為什麼為static型別就
會解決這個問題呢?因為在java中所有非靜態的物件都會持有當前類的強引用,而靜態物件則只會持有當前類的弱引用。宣告為靜態後,handler將會持
有一個Activity的弱引用,而弱引用會很容易被gc回收,這樣就能解決Activity結束後,gc卻無法回收的情況。(至於為什麼強引用不能夠被gc自動回收,而
弱引用物件為什麼會被gc回收,可以自行去google)。
所以解決這個警告就有幾種方法:
一:將hanlder物件宣告為靜態的物件。
二:使用靜態內部類,通過WeakReference實現對Activity的弱引用。具體實現看以下程式碼:
public classAutoActivityextendsActivity{
MyHandler handler = new MyHandler(this);
@Override
protectedvoidonCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_auto);
}
static classMyHandlerextendsHandler{
WeakReference<AutoActivity> mactivity;
publicMyHandler(AutoActivity activity){
mactivity = new WeakReference<AutoActivity>(activity);
}
@Override
publicvoidhandleMessage(Message msg){
super.handleMessage(msg);
switch (msg.what) {
case 100:
//在這裡面處理msg
//通過mactivity.get()獲取Activity的引用(即上下文context)
break;
default:
break;
}
}
}
}
來自:https://my.oschina.net/u/1177694/blog/523922