溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

由單例模式造成的內(nèi)存泄漏

發(fā)布時間:2020-06-26 22:46:48 來源:網(wǎng)絡(luò) 閱讀:3105 作者:whatever957 欄目:移動開發(fā)


使用單例模式時,有時候不小心,就會很容易造成內(nèi)容泄漏,如下代碼所示:

public class SingleInstance
{
private static volatile SingleInstance instance;
private Context context;
private SingleInstance(Context context)
{
    this.context = context;
}
public static SingleInstance getInstance(Context context)
{
    if(instance == null)
    {
        synchronized(SingleInstance.class)
        {
            if(instance == null)
            {
            instance = new SingleInstance(context);
            }
        }
    }
    return instance;
    }
}
public class MainActivity extends Activity
{
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //leak occured
        SingleInstance.getInstance(this);
    }
}

上面的代碼中,傳入給單例對象的context是Activity的context,而單例對象是一個static對象,其生命周期與應(yīng)用程序是一致的,(也就是說,只有應(yīng)用程序進程被殺掉了,static對象才會被銷毀,因為static是類對象,而不是對象變量),該SingleInstance單例靜態(tài)對象持有當前Activity的context,當MainActivity退出時,由于instance還繼續(xù)只有其context引用,對造成系統(tǒng)無法銷毀該Activity,從而造成內(nèi)存泄漏。


解決方法:

從以上分析中,可以看成,造成內(nèi)存泄漏的主要原因就是static對象的生命周期與其持有對象引用(即Activity)的聲明周期不同而造成的,因此,解決內(nèi)存的泄漏的方法有如下2種:

  1. 使用應(yīng)用程序的getApplicationContext(),靜態(tài)對象的生命周期與應(yīng)用程序的生命周期一致,故此不會導(dǎo)致內(nèi)存泄漏。

  2. 持有傳入的context的弱引用。如下所示:

private WeakReference<Context> weakContext;
private SingleInstance(Context context)
{
    weakContext = new WeakReference<Context>(context);
}

如果某個時間點,MainActivity被GC了,由于持有的是MainActivity的弱引用,不會影響系統(tǒng)對MainActivity的回收,那么context就被置空了,所以后面要使用該context時,就需要判斷一下該若引用持有的對象是否還存在:

weakContext.get() != null


向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI