案例:(本篇還以上篇案例為例子)
public class CommUtil {
private static CommUtil instance;
private Context context;
private CommUtil(Context context) {
this.context=context;
}
public static CommUtil getInstance(Context context){
if(null==instance){
instance=new CommUtil(context);
}
return instance;
}
}
複製程式碼
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
CommUtil instance = CommUtil.getInstance(this);
}
}
複製程式碼
(將Activity的例項被一個單例物件所持有,在旋轉螢幕的時候造成記憶體洩漏) 將專案執行起來 開啟Android Profiler 觀察Memory 並且試著多次旋轉螢幕就會發現記憶體一直在增高
data:image/s3,"s3://crabby-images/a19ba/a19ba05e389dbd5e668d6332b1fae8c7d546376c" alt="Andriod專案記憶體洩漏流程"
data:image/s3,"s3://crabby-images/8dcc5/8dcc5ed0994dce13be9d5f9e8415980b47f09f84" alt="Andriod專案記憶體洩漏流程"
Dump java Heap: 點選就會生成app執行記憶體快照.hprof檔案 然後將APP完全退出 重新啟動 開啟Android Monitor 點選Dump java Heap點選生成一份還沒操作前(旋轉螢幕)的記憶體快照hprof檔案(為以後作對比用) 現在就已經生成好了2份hprof檔案 一份是沒有旋轉過螢幕的 一份是旋轉過螢幕多次的 然後選中Android Studio 最左邊的Captures 進行將hprof檔案匯出(因為MAT不支援Android Studio生成的未經轉換的hprof檔案 在匯出過程中Android Studio會為我們轉換好)
data:image/s3,"s3://crabby-images/94aac/94aac21378576cbe209ad4caed1138e290a3d82b" alt="Andriod專案記憶體洩漏流程"
data:image/s3,"s3://crabby-images/46f14/46f141de7ebf96c06288970a91bd66ea4ae25a23" alt="Andriod專案記憶體洩漏流程"
MAT上場了
安裝 Memory Analyzer 外掛
data:image/s3,"s3://crabby-images/fd489/fd48938e6481a4bfc84e1a2a3159a88b9492421e" alt="Andriod專案記憶體洩漏流程"
data:image/s3,"s3://crabby-images/48a0b/48a0bc80999ce47a34a915d0efcea27ea596a346" alt="Andriod專案記憶體洩漏流程"
匯入生成的兩個檔案
data:image/s3,"s3://crabby-images/72c18/72c18d9c81901ac09cce07fc0dea7905c00f7fa6" alt="Andriod專案記憶體洩漏流程"
這個時候可能會報錯
Unknown HPROF Version (JAVA PROFILE 1.0.3) (java.io.IOException) 這是應為studio生成的.hprof檔案和eclipse支援的.hprof檔案格式問題
轉換格式: 進入SDK目錄輸入 hprof-conv inputfilepath outputfilepath之後從新匯入即可
先來看看MAT都有哪些功能:
Overview檢視 該檢視會首頁總結出當前這個Heap dump佔用了多大的記憶體,其中涉及的類有多少,物件有多少,類載入器,如果有沒有回收的物件,會有一個連線,可以直接參看(圖中的Unreachable Objects Histogram)。
data:image/s3,"s3://crabby-images/ac605/ac60587ba15194318ca3841efba849281f5974a4" alt="Andriod專案記憶體洩漏流程"
histogram檢視: 列舉記憶體中物件存在的個數和大小 Dominator tree檢視:該檢視會以佔用總記憶體的百分比來列舉所有例項物件,注意這個地方是物件而不是類了,這個檢視是用來發現大記憶體物件的 Top Consumers: 該檢視會顯示可能的記憶體洩漏點 Duplicate Classes: 該檢視顯示重複的類等資訊 本次重點圍繞histogram檢視講解 其他3個檢視也只是給我提供一些可能會造成記憶體洩漏的資訊 點選開啟Histograms檢視
data:image/s3,"s3://crabby-images/84d42/84d42d39b13c15c838a555eef79e9d19e4c99fd5" alt="Andriod專案記憶體洩漏流程"
開啟下面的皮膚Navigation History 選中histogram右鍵add to Compare Basket 新增到比較容器中(2個hprof檔案都做以上同樣的操作 將histogram 新增到比較容器中)
data:image/s3,"s3://crabby-images/930a2/930a202c08be545f0d6e20f2bf77b9f34bff4141" alt="Andriod專案記憶體洩漏流程"
這時比較容器中就有2份hprof檔案 一份未做任何操作 一份做了多次旋轉螢幕(注意:匯入的比較容器的順序很重要 不然後面的資料看起來讓人費解)然後點選Compare the results (紅色感嘆號)進行比較
data:image/s3,"s3://crabby-images/c0065/c0065048d4a36a30eadbaab600be532637876715" alt="Andriod專案記憶體洩漏流程"
通過比較後就會生成一個比較結果表ComPared Tables
data:image/s3,"s3://crabby-images/a8cba/a8cba013b8ae15286e7f01dee7542c0946a45805" alt="Andriod專案記憶體洩漏流程"
但是這個表內容太多 如何快速過濾出我們與我們自己寫的專案內容有關的呢 接著在Class Name 輸入我們的專案包名
data:image/s3,"s3://crabby-images/0c51f/0c51f7ca74fb3e7ff81ff0b7130c8d6dcce80e9f" alt="Andriod專案記憶體洩漏流程"
通上圖就明顯看出經過2個hprof比較 在我們操作了旋轉多次螢幕後MainActivity的例項增加了然後我們就去分析下該hprof檔案是什麼導致MainActivity記憶體洩漏了
data:image/s3,"s3://crabby-images/9a881/9a881933949d24fb00c485886201ffed279b7122" alt="Andriod專案記憶體洩漏流程"
在Class Name處可以根據輸入的類名去查詢對應的物件例項(比如我們輸入MainActivty):
data:image/s3,"s3://crabby-images/b51bd/b51bdc03fe9ee5020881be9944aed590f4b60510" alt="Andriod專案記憶體洩漏流程"
Objects:例項個數 Shallow Heap:所佔記憶體大小 Retained Heap:釋放後能回收多少記憶體 通過上圖看到了MainActivtiy在旋轉螢幕後產生了2個例項 再去觀察MainActivtiy具體被哪些物件引用呢 滑鼠選中MainActivtiy 例項右鍵 選擇with incoming references:
data:image/s3,"s3://crabby-images/a73d5/a73d56c0a02eff2347b88c6d88889ffbf35b0686" alt="Andriod專案記憶體洩漏流程"
data:image/s3,"s3://crabby-images/bf60b/bf60b95e244fdbe4881af974cf1a805eb6154207" alt="Andriod專案記憶體洩漏流程"
data:image/s3,"s3://crabby-images/83a97/83a97aebe2129e13104d6c2c6068d087f99e0f3d" alt="Andriod專案記憶體洩漏流程"
看到MainActivty被引用的地方這麼多 而且一屏還顯示不完 我們又如何去判斷是哪個導致記憶體洩漏的呢 MAT還有一個功能 就是通過遍歷GC Root樹去將那些有可能被GC回收的例項 將他們去除(備註:在GC Root樹中能找到的物件絕對不存在有記憶體洩漏的例項 因為他們在執行時會被回收的嘛 只有找不到的那些才是)滑鼠右鍵:
data:image/s3,"s3://crabby-images/c5add/c5add240564bfcf59e7c5bacb51f881b786e65ec" alt="Andriod專案記憶體洩漏流程"
data:image/s3,"s3://crabby-images/71908/71908ed3173acde181df809aeaa0c9445351c3f2" alt="Andriod專案記憶體洩漏流程"
data:image/s3,"s3://crabby-images/4e342/4e342bea47e4b6775cbef53b792336e2ba47060f" alt="Andriod專案記憶體洩漏流程"
看那些單詞就知道是什麼意思 (排除 軟引用 弱引用 虛引用) 最後我們就看到了Commutil的instance引用了MainActivty例項 而後旋轉螢幕的時候 系統又建立了一個 所以有2個引用著MainActivtiy的例項 造成的記憶體洩漏:
然後就去修改程式碼: public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
CommUtil instance = CommUtil.getInstance(getApplicationContext());
}
}
複製程式碼