前言:
1.該內容為APP應用用戶端的效能測試,未涉及後台,所以並非針對API或資料介面
2.測試的目標項:資源消耗、記憶體泄露、電量功耗、回應時間
3.用戶端的效能指標:記憶體、CPU、流量
4.本系列主要是講述 如何擷取安卓APP應用的效能指標,並簡單分析,定位問題
一、查看 記憶體 指標:
準備工作:
(1).進入裝有測試APP手機的 “開發人員選項” 並開啟“USB偵錯模式”
(2).使用資料線將手機裝置與電腦裝置串連,可裝PP助手進行接入
(3).手機開啟待測APP,即開啟進程
1.命令列查看記憶體資料:
(1).開啟cmd
(2). 擷取裝置列表:輸入 adb devices(預先安裝adb驅動、若報錯,拔掉重新串連手機)
(3).進入該裝置的shell環境:輸入:adb -s C7R6T16722004661 shell (若只有一台裝置,可直接 adb shell ,多台必須加裝置序號C7R6T16722004661)
(4).尋找進程:輸入ps (模糊尋找) 尋找對應待測應用程式套件名,並記錄下其的pid(進程ID):30017,
若知道明確的包名,可直接準確尋找 ps |grep com.hundsun.stockwinner.grzq
(5).查詢記憶體資訊:
可通過兩種方式擷取:
a.通過 “ dumpsys meminfo 包名/pid ” 命令擷取,輸入:dumpsys meminfo 30017
Pss Total :實際使用的實體記憶體
private dirty:私人駐留記憶體
Heap Size: 佔用總記憶體(Heap 堆)(擴充:進程記憶體空間是虛擬記憶體,區分於實體記憶體,進程無法直接操作實體記憶體RAM。必要時,作業系統對其進行映射,使進程能應用到實體記憶體)
Heap Alloc: 分配記憶體
Heap Free: 空閑記憶體
native process和java process佔據記憶體的大小(擴充:c++申請的記憶體為native process,java申請的記憶體:java process)
記憶體大小:native process:13004
dalvik process:10448
註:因為Android系統對dalvik的vm heapsize作了硬性限制,當java進程申請的java空間超過閾值時,就會拋出OOM異常(這個閾值可以是48M、24M、16M等,視機型而定)
查看單個應用最大記憶體限制,輸入命令:getprop|grep heapgrowthlimit 得到結果該機型為192M。dalvik process 超過就會拋OOM異常
b.可直接通過:procrank 。
手機中的sh是經過精簡過的,有些手機可能沒有 procrank 命令,可以使用genymotion模擬器,或是自己安裝procrank命令。
(我也沒這個命令,沒有裝好,這部分內容未操作,為網路直接擷取)
VSS- Virtual Set Size 虛擬耗用記憶體(包含共用庫佔用的記憶體)——是單個進程全部可訪問的地址空間
RSS- Resident Set Size 實際使用實體記憶體(包含共用庫佔用的記憶體)——單個進程實際佔用的記憶體大小,不是精確描述
PSS- Proportional Set Size 實際使用的實體記憶體(比例分配共用庫佔用的記憶體)——對於系統中的整體記憶體使用量是一個很好的描述
USS- Unique Set Size 進程獨自佔用的實體記憶體(不包含共用庫佔用的記憶體)——單個進程的全部私人記憶體大小,亦即全部被該進程獨佔的記憶體大小。
一般來說記憶體佔用大小有如下規律:VSS >= RSS >= PSS >= USS。
USS 是針對某個進程開始有可疑記憶體泄露的情況,進行檢測的最佳數字
2.代碼擷取記憶體資料:
java調用Adb shell dumpsys meminfo再用字串截取方式擷取記憶體,可對其資料進行返回列印,實現監控。
傳入參數為:應用程式套件名,這個不可變。
package com.hss.performanceTest;/** * from hss * data:2017/9/8 * project:getMemory */import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;public class getMemory { public static String GetMemory(String packageName) throws IOException, InterruptedException { String str3=null; Runtime runtime = Runtime.getRuntime(); Process proc = runtime.exec("adb shell dumpsys meminfo "+packageName); try { if (proc.waitFor() != 0) { System.err.println("exit value = " + proc.exitValue()); } BufferedReader in = new BufferedReader(new InputStreamReader( proc.getInputStream())); StringBuffer stringBuffer = new StringBuffer(); String line = null; while ((line = in.readLine()) != null) { stringBuffer.append(line+" "); } String str1=stringBuffer.toString(); String str2=str1.substring(str1.indexOf("Objects")-60,str1.indexOf("Objects")); str3=str2.substring(0,10); str3.trim(); } catch (InterruptedException e) { System.err.println(e); }finally{ try { proc.destroy(); } catch (Exception e2) { } } return str3 ; } public static void main(String args[]) { System.out.println("開始運行..."); try {String resurt = getMemory.GetMemory("com.hundsun.stockwinner.sxzq");System.out.println("山西證券的記憶體:"+resurt);} catch (IOException e) {// TODO Auto-generated catch blocke.printStackTrace();} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();} } }
運行截圖: