android中常見(jiàn)的內(nèi)存泄露

使用工具:

可使用android studio工具進(jìn)行監(jiān)測(cè):

還可以使用LeakCanary和MAT進(jìn)行分析;

常見(jiàn)的泄露:

Context:在項(xiàng)目中遇到context泄露的問(wèn)題,當(dāng)activity的context被static的類(lèi)給強(qiáng)引用了導(dǎo)致activity不能被回收,有一個(gè)辦法就是設(shè)置軟引用,但這也不可,因?yàn)楹芸赡軙?huì)報(bào)空指針錯(cuò)誤;所以可以使用ApplicationContext,因?yàn)锳pplicationContext的生命周期也是伴隨進(jìn)程的不需要被回收。但是不能都使用applicationcontext,因?yàn)橛行﹫?chǎng)景是不能用的,例如需要涉及UI的。因此可以盡量使用applicationcontext,在涉及UI的部分就使用Activity Context并且注意內(nèi)存泄露。
下面引用一下鴻洋大神博客的一張圖:

使用Handler時(shí)要注意內(nèi)存泄露的可能:因?yàn)閔andler被創(chuàng)建實(shí)例后會(huì)被發(fā)送到消息隊(duì)列的message擁有一個(gè)強(qiáng)引用,當(dāng)message一直存在于消息隊(duì)列時(shí),handler就不會(huì)被回收;如果它是一個(gè)匿名類(lèi)或者是非靜態(tài)內(nèi)部類(lèi)時(shí),那么就會(huì)持有外部類(lèi)的引用導(dǎo)致外部類(lèi)不能被回收;
解決方法:可以設(shè)置Handler為靜態(tài)內(nèi)部類(lèi),當(dāng)需要外部類(lèi)時(shí),將外部類(lèi)設(shè)置為弱引用;
如下引用別人代碼:

public class SampleActivity extends Activity {

  /**
   * Instances of static inner classes do not hold an implicit
   * reference to their outer class.
   */
  private static class MyHandler extends Handler {
    private final WeakReference<SampleActivity> mActivity;

    public MyHandler(SampleActivity activity) {
      mActivity = new WeakReference<SampleActivity>(activity);
    }

    @Override
    public void handleMessage(Message msg) {
      SampleActivity activity = mActivity.get();
      if (activity != null) {
        // ...
      }
    }
  }

  private final MyHandler mHandler = new MyHandler(this);

  /**
   * Instances of anonymous classes do not hold an implicit
   * reference to their outer class when they are "static".
   */
  private static final Runnable sRunnable = new Runnable() {
      @Override
      public void run() { /* ... */ }
  };

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    // Post a message and delay its execution for 10 minutes.
    mHandler.postDelayed(sRunnable, 1000 * 60 * 10);

    // Go back to the previous Activity.
    finish();
  }
}

使用HandlerThead時(shí)也要注意:因?yàn)镠andlerThread實(shí)現(xiàn)的run方法是一個(gè)無(wú)限循環(huán),它不會(huì)自己結(jié)束,線(xiàn)程的生命周期超過(guò)了activity生命周期,當(dāng)橫豎屏切換,HandlerThread線(xiàn)程的數(shù)量會(huì)隨著activity重建次數(shù)的增加而增加。
應(yīng)該在onDestroy時(shí)將線(xiàn)程停止掉:mThread.getLooper().quit();
另外,對(duì)于不是HandlerThread的線(xiàn)程,也應(yīng)該確保activity消耗后,線(xiàn)程已經(jīng)終止,可以這樣做:在onDestroy時(shí)調(diào)用mThread.join();

項(xiàng)目中的遇到的知識(shí)點(diǎn)和問(wèn)題總結(jié)(備注):

在我的項(xiàng)目中就出現(xiàn)了使用context被static對(duì)象引用導(dǎo)致不能回收
靜態(tài)內(nèi)部類(lèi)的對(duì)象:首先多次new是在堆中創(chuàng)建了不同的對(duì)象,和普通類(lèi)的對(duì)象一樣。我在使用的時(shí)候,主要怕多個(gè)靜態(tài)內(nèi)部類(lèi)的對(duì)象在程序運(yùn)行過(guò)程中像靜態(tài)成員變量一樣一直占據(jù)內(nèi)存空間導(dǎo)致OOM,經(jīng)過(guò)驗(yàn)證,靜態(tài)內(nèi)部類(lèi)的對(duì)象雖然每次new都生成一個(gè)對(duì)象,但是會(huì)被及時(shí)的回收,不會(huì)因?yàn)橐恢闭紦?jù)內(nèi)存空間導(dǎo)致OOM。
$后面跟數(shù)字是匿名類(lèi)編譯出來(lái)的
$后面跟文字是內(nèi)部類(lèi)編譯出來(lái)的

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

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

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