成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Android性能優化之內存泄漏

移動開發 Android
內存泄漏(memory leak)是指由于疏忽或錯誤造成程序未能釋放已經不再使用的內存。那么在Android中,當一個對象持有Activity的引用,如果該對象不能被系統回收,那么當這個Activity不再使用時,這個Activity也不會被系統回收,那這么以來便出現了內存泄漏的情況。

綜述

內存泄漏(memory leak)是指由于疏忽或錯誤造成程序未能釋放已經不再使用的內存。那么在Android中,當一個對象持有Activity的引用,如果該對象不能被系統回收,那么當這個Activity不再使用時,這個Activity也不會被系統回收,那這么以來便出現了內存泄漏的情況。在應用中內出現一次兩次的內存泄漏獲取不會出現什么影響,但是在應用長時間使用以后,若是存在大量的Activity無法被GC回收的話,最終會導致OOM的出現。那么我們在這就來分析一下導致內存泄漏的常見因素并且如何去檢測內存泄漏。

導致內存泄漏的常見因素

情景一:靜態Activity和View

靜態變量Activity和View會導致內存泄漏,在下面這段代碼中對Activity的Context和TextView設置為靜態對象,從而產生內存泄漏。

  1. import android.content.Context; 
  2. import android.support.v7.app.AppCompatActivity; 
  3. import android.os.Bundle; 
  4. import android.widget.TextView; 
  5.   
  6. public class MainActivity extends AppCompatActivity { 
  7.   
  8.     private static Context context; 
  9.     private static TextView textView; 
  10.   
  11.     @Override 
  12.     protected void onCreate(Bundle savedInstanceState) { 
  13.         super.onCreate(savedInstanceState); 
  14.         setContentView(R.layout.activity_main); 
  15.         context = this; 
  16.         textView = new TextView(this); 
  17.     } 

 

情景二:Thread,匿名類,內部類

在下面這段代碼中存在一個非靜態的匿名類對象Thread,會隱式持有一個外部類的引用LeakActivity,從而導致內存泄漏。同理,若是這個Thread作為LeakActivity的內部類而不是匿名內部類,他同樣會持有外部類的引用而導致內存泄漏。在這里只需要將為Thread匿名類定義成靜態的內部類即可(靜態的內部類不會持有外部類的一個隱式引用)。

  1. public class LeakActivity extends AppCompatActivity { 
  2.   
  3.     @Override 
  4.     protected void onCreate(Bundle savedInstanceState) { 
  5.         super.onCreate(savedInstanceState); 
  6.         setContentView(R.layout.activity_leak); 
  7.         leakFun(); 
  8.     } 
  9.   
  10.     private void leakFun(){ 
  11.         new Thread(new Runnable() { 
  12.             @Override 
  13.             public void run() { 
  14.                 try { 
  15.                     Thread.sleep(Integer.MAX_VALUE); 
  16.                 } catch (InterruptedException e) { 
  17.                     e.printStackTrace(); 
  18.                 } 
  19.             } 
  20.         }); 
  21.     } 
  22.  

情景三:動畫

在屬性動畫中有一類無限循環動畫,如果在Activity中播放這類動畫并且在onDestroy中去停止動畫,那么這個動畫將會一直播放下去,這時候Activity會被View所持有,從而導致Activity無法被釋放。解決此類問題則是需要早Activity中onDestroy去去調用objectAnimator.cancel()來停止動畫。

  1. public class LeakActivity extends AppCompatActivity { 
  2.   
  3.     private TextView textView; 
  4.     @Override 
  5.     protected void onCreate(Bundle savedInstanceState) { 
  6.         super.onCreate(savedInstanceState); 
  7.         setContentView(R.layout.activity_leak); 
  8.         textView = (TextView)findViewById(R.id.text_view); 
  9.         ObjectAnimator objectAnimator = ObjectAnimator.ofFloat(textView,"rotation",0,360); 
  10.         objectAnimator.setRepeatCount(ValueAnimator.INFINITE); 
  11.         objectAnimator.start(); 
  12.     } 
  13.  

情景四:Handler

對于Handler的內存泄漏在(Android的消息機制——Handler的工作過程)[http://blog.csdn.net/ljd2038/article/details/50889754]這篇文章中已經詳細介紹,就不在贅述。

情景五:第三方庫使用不當

對于EventBus,RxJava等一些第三開源框架的使用,若是在Activity銷毀之前沒有進行解除訂閱將會導致內存泄漏。

使用MAT檢測內存泄漏

對于常見的內存泄露進行介紹完以后,在這里再看一下使用MAT(Memory Analysis Tool)來檢測內存泄露。MAT的下載地址為:http://www.eclipse.org/mat/downloads.php。

下面來看一段會導致內存泄露的錯誤代碼。

  1. public class LeakActivity extends AppCompatActivity { 
  2.   
  3.     @Override 
  4.     protected void onCreate(Bundle savedInstanceState) { 
  5.         super.onCreate(savedInstanceState); 
  6.         setContentView(R.layout.activity_leak); 
  7.         EventBus.getDefault().register(this); 
  8.     } 
  9.   
  10.     @Subscribe 
  11.     public void subscriber(String s){ 
  12.   
  13.     } 
  14.  

在上面這段代碼中有會導致內存泄漏,原因是EventBus沒有解除注冊。下面就以這段代碼為例來看一下如何分析內存泄漏。

打開AndroidStudio中的Monitors可以看到如下界面。   

打開AndroidStudio中的Monitors 

在這里可以看到在應用剛啟動的時候,所占用的內存為15M,然后我們現在開始操作APP,反復進入退出LeakActicity。點擊上如中的GC按鈕。這時候我們在看一下內存使用情況。 

 

 

內存使用情況 

在這里我們可以看到,內存一直在持續增加,已經達到33M,并且無法被GC所回收。所以我們可以判斷,這時候必然出現內存泄漏的情形。那么現在再點擊Dump Java Heap按鈕,在captures窗口看到生成得hprof文件。但這時候所生成的hprof文件不是標準格式的,我們需要通過SDK所提供的工具hprof-conv進行轉化,該工具在SDK的platform-tools目錄下。執行命令如下:

  1. hprof-conv XXX.hprof converted-dump.hprof 

當然在AndroidStudio中可以省去這一步,可以直接導出標準格式的hprof文件。 

 

 

直接導出標準格式的hprof文件 

這時候可以通過MAT工具來打開導出的hprof文件。打開界面如下圖所示: 

 

 

可以通過MAT工具來打開導出的hprof文件 

在MAT中我們最常用的就是Histogram和Dominator Tree,他們分別對應上圖中的A和B按鈕。Histogram可以看出內存中不同類型的buffer的數量和占用內存的大小,而Dominator Tree則是把內存中的對象按照從大到小的順序進行排序,并且可以分析對象之間的引用關系。在這里再來介紹一下MAT中兩個符號的含義。

  • ShallowHeap:對象自身占用的內存大小,不包括他引用的對象
  • RetainedHeap:對象自身占用的內存大小并且加上它直接或者間接引用對象的大小

Histogram

由于在Android中一般內存泄漏大多出現在Acivity中,這時候可以點擊Histogram按鈕,并搜索Activity。   

點擊Histogram按鈕,并搜索Activity 

在這里可以看出LeakActivity存在69個對象,基本上可以斷定存在內存泄漏的情形,這時候便可以通過查看GC對象的引用鏈來進行分析。點擊鼠標右鍵選擇Merge Shortest paths to GC Roots并選擇exclude weak/soft references來排除弱引用和軟引用。 

 

 

點擊鼠標右鍵選擇Merge Shortest paths to GC Roots并選擇exclude weak/soft references來排除弱引用和軟引用。

 

在排除軟引用和弱引用以后如下圖所示: 

 

 

在排除軟引用和弱引用以后 

在這里可以看出由于EventBus導致的LeakActivity內存泄漏。

在Histogram中還可以查看一個對象包含了那些對象的引用。例如,現在要查看LeakActivity所包含的引用,可以點擊鼠標右鍵,選擇list objects中的with incoming reference。而with outcoming reference表示選中對象持有那些對象的引用。 

 

 

點擊鼠標右鍵,選擇list objects中的with incoming reference

 

Dominator Tree

現在我們點擊這時候可以點擊Dominator Tree按鈕,并搜索Activity。可以看到如下圖所示: 

 

 

點擊Dominator Tree按鈕,并搜索Activity

 

在這里可以看到存在大量的LeakActivity。然后點擊鼠標右鍵選擇Path To GC Roots->exclude weak/soft references來排除弱引用和軟引用。 

 

 

點擊鼠標右鍵選擇Path To GC Roots->exclude weak/soft references來排除弱引用和軟引用 

之后可以看到如下結果,依然是EventBus導致的內存泄漏: 

 

 

EventBus導致的內存泄漏 

總結

內存泄漏往往被我們所忽略,但是當大量的內存泄漏以后導致OOM。它所造成的影響也是不容小覷的。當然除了上述內存泄漏的分析以為我們還可以通過LeakCanary來分析內存泄漏。對于LeakCanary的使用在這里就不在進行詳細介紹。

責任編輯:龐桂玉 來源: 安卓開發精選
相關推薦

2015-09-16 15:21:23

Android性能優化內存

2017-03-14 18:48:06

Android性能優化內存優化

2011-04-25 14:06:23

java

2011-12-28 13:38:00

JavaJVM

2023-10-31 16:40:38

LeakCanary內存泄漏

2023-06-12 00:22:50

操作系統應用程序內核鎖

2011-06-20 13:35:01

網站優化

2016-03-21 10:31:25

Android內存泄露

2013-08-07 10:16:43

Android內存泄漏

2017-12-14 14:32:30

.Net內存代碼

2016-07-05 14:09:02

AndroidJAVA內存

2014-07-30 14:22:41

AndroidWebView內存泄漏

2011-06-01 12:50:41

Android 內存

2023-02-20 15:27:30

開發JavaScript內存管理

2013-09-16 15:16:20

Android性能優化

2022-09-09 15:58:29

HiveServerHive 組件Java 開發

2021-07-29 14:20:34

網絡優化移動互聯網數據存儲

2024-07-03 11:28:15

2024-03-11 08:22:40

Java內存泄漏

2023-12-18 10:45:23

內存泄漏計算機服務器
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩国产高清在线观看 | 国产探花在线精品一区二区 | 欧美精品第三页 | 日韩欧美一级片 | 国产亚洲一区精品 | 天天干天天草 | 久久精品视频一区二区三区 | 精品国产一区二区三区久久狼黑人 | 亚洲成人精品一区 | 一区二区免费 | 亚洲精品久久久一区二区三区 | 午夜影院视频 | 久久久久久高潮国产精品视 | 精品欧美乱码久久久久久 | 久久国产精品一区二区三区 | h视频免费在线观看 | www.99热这里只有精品 | 欧美福利| 激情三区| 殴美成人在线视频 | 五月天天丁香婷婷在线中 | 国产成人精品在线 | 成人精品在线观看 | 日批免费观看 | 国产精品区二区三区日本 | 四虎影院在线观看av | 国产在线a| 精品无码久久久久久国产 | 91久久国产精品 | 亚洲精品久久久久久久久久久 | 亚洲一区二区三区观看 | 中文字幕一区在线观看视频 | 日韩在线一区二区三区 | 精品婷婷 | 日韩av中文 | 美国a级毛片免费视频 | 久久久久亚洲 | 欧美黄在线观看 | 四虎伊人 | 欧美精品久久久 | 日韩欧美视频免费在线观看 |