Android中如何獲得系統(tǒng)相關的信息(比如CPU使用率,內存的總量和已用的量)
成都創(chuàng)新互聯(lián)公司-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設、高性價比開魯網(wǎng)站開發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫,直接使用。一站式開魯網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設找我們,業(yè)務覆蓋開魯?shù)貐^(qū)。費用合理售后完善,10余年實體公司更值得信賴。
1、查看CPU
我想大家都很關系自己的愛機的CUP到底是什么具體信息呢.那就跟我來操縱吧.
打開超級終端,在命令行下輸入SU回車,然后在#后輸入cat /proc/cpuinfo回車.看到了吧.一串信息顯示出來了.具體如下(以本人手機為例):
# cat cpuinfo
cat cpuinfo
Processor : ARMv6-compatible processor rev 2 (v6l)
BogoMIPS : 527.15
Features : swp half thumb fastmult edsp java
CPU implementer : 0x41
CPU architecture: 6TEJ
CPU variant : 0x1
CPU part : 0xb36
CPU revision : 2
Hardware : trout
Revision : 0080
Serial : 0000000000000000
里面的Processor 是處理器的意思.本人處理器是ARMv6兼容處理器
BogoMIPS(讀作bogumips)是Linux操作系統(tǒng)中衡量計算機處理器運行速度的的一種尺度,而提供這種度量的程序也被稱為BogoMips,是由Linux主要開發(fā)者linus Torvalds寫的。
本人的是527.15
2、查看內存
內存查看可以從設置--存儲卡和手機存儲--里面會可以看到也可以下載高級任務管理器,方便你關閉程序和查詢內存
3、通過查看文件獲取設備信息
通過讀取文件/proc/meminfo的信息獲取Memory的總量。
ActivityManager. getMemoryInfo(ActivityManager.MemoryInfo)獲取當前的可用Memory量。
通過讀取文件/proc/cpuinfo系統(tǒng)CPU的類型等多種信息。
讀取/proc/stat 所有CPU活動的信息來計算CPU使用率
命令中的"應用包名"應該替換為你需要查詢的包名.
執(zhí)行命令后, 在輸出的內容中, 第二項即應用的進程名, 例如:
那么 22411 即為該應用當前的pid.
其中的"應用的pid"為上一步獲取到的pid
執(zhí)行命令后, 命令行工具即會打印應用運行信息
命令中的"應用包名"應該替換為你需要查詢的包名.
命令執(zhí)行后過段時間即會打印內存占用大小.
3)查看進程列表:adb shell "ps",同時也能獲取到應用的UID,方式如下(不需root權限):
u0_a開頭的都是Android的應用進程,Android的應用的UID是從10000開始,到19999結束,可以在Process.java中查看到(FIRST_APPLICATION_UID和LAST_APPLICATION_UID),u0_a后面的數(shù)字就是該應用的UID值減去FIRST_APPLICATION_UID所得的值,所以,對于截圖這個應用進程,它是u0_a155,按前面的規(guī)制,它的UID就是155 + FIRST_APPLICATION_UID = 10155。
VSS - Virtual Set Size 虛擬耗用內存(包含共享庫占用的內存)
RSS - Resident Set Size 實際使用物理內存(包含共享庫占用的內存)
PSS - Proportional Set Size 實際使用的物理內存(比例分配共享庫占用的內存)
USS - Unique Set Size 進程獨自占用的物理內存(不包含共享庫占用的內存)
一般來說內存占用大小有如下規(guī)律:VSS = RSS = PSS = USS
使用 adb shell "dumpsys meminfo -s pakagename | pid"命令,輸出結果分以下4部分:
PS:在apk內調用運行獲取其他app的內存數(shù)據(jù)則需要root權限
adb命令:adb shell dumpsys gfxinfo package | pid
正常情況下幀率應該在16.67ms左右,1秒60幀,執(zhí)行結果如下:
詳細計算方法如下:
還有一個命令是: adb shell dumpsys SurfaceFlinger --latency LayerName
其中LayerName在各個不同系統(tǒng)中獲取的命令是不一樣的
在Android 6系統(tǒng)直接就是SurfaceView
在Android 7系統(tǒng)中可以通過 dumpsys window windows | grep mSurface | grep SurfaceView 然后通過數(shù)據(jù)截取到
在Android 8系統(tǒng)中可以通過 dumpsys SurfaceFlinger | grep android包名獲取到
執(zhí)行命令結果如下:
計算方法比較簡單,一般打印出來的數(shù)據(jù)是129行(部分機型打印兩次257行,但是第一部分是無效數(shù)據(jù),取后半部分),取len-2的第一列數(shù)據(jù)為end_time,取len-128的第一列數(shù)據(jù)為start_time
fps = 127/((end_time - start_time) / 1000000.0)
至于為啥要取第一列數(shù)據(jù),這里不做過多介紹,歡迎參看這兩篇文章
老羅的文章SurfaceView原理
Android性能測試之fps獲取
至于為啥要處于1000000,因為命令打印出來的是納秒單位,要轉為毫秒進行計算,127就是因為命令一次打印出來127幀的數(shù)據(jù)而已
有兩種方法可以獲取
1) adb shell "top -n 5 | grep package | pid" ,第三列就是實時監(jiān)控的CPU占用率(-n 指定執(zhí)行次數(shù),不需root權限),這邊top命令執(zhí)行需要2到3s左右,一般可以采用busybox 的top命令執(zhí)行,效率會快很多
2) adb shell "dumpsys cpuinfo | grep package | pid"
兩種方法直接區(qū)別在于,top是持續(xù)監(jiān)控狀態(tài),而dumpsys cpuinfo獲取的實時CPU占用率數(shù)據(jù)
adb命令:adb shell "dumpsys batterystats package | pid" (Android 5.0后引入)
獲取單個應用的耗電量信息,具體返回結果待研究
adb命令:adb shell "dumpsys battery"
出現(xiàn)信息解讀:
AC powered:false 是否連接AC(電源)充電線
USB powered:true 是否連接USB(PC或筆記本USB插口)充電
Wireless powered:false 是否使用了無線電源
status: 1 電池狀態(tài),2為充電狀態(tài),其他為非充電狀態(tài)
level:58 電量(%)
scale: 100. 電量最大數(shù)值
voltage: 3977 當前電壓(mV)
current now: -335232. 當前電流(mA)
temperature:355 電池溫度,單位為0.1攝氏度
adb 命令:adb shell "dumpsys package | pid | grep UID" [通過ps命令,獲取app的UID(安裝后唯一且固定)]
adb shell cat /proc/uid_stat/UID/tcp_rcv [cat為查看命令,讀取tcp_rcv獲取應用接收流量信息(設備重啟后清零)]
adb shell cat /proc/uid_stat/UID/tcp_snd [cat為查看命令,讀取tcp_snd獲取應用發(fā)送流量信息(設備重啟后清零)]
計算流量消耗步驟:
或者還有一種方式獲取應用流量消耗:
首先判斷類型:
cat /sys/class/thermal/thermal_zone*/type
只有紅框框出來的是有效的
cat /sys/class/thermal/thermal_zone*/temp
獲取CPU溫度
dumpsys battery | grep temperature 單位0.1攝氏度
獲取/proc/stat文件內容(無權限限制)
總的cpu時間片是 total = user+nice+system+idle+iowait+irq+softirq
忙碌時間為 notidle = user+nice+system +iowait+irq+softirq
cpu使用率計算方法為,先取開始的total值和忙碌時間notidle,隔一段時間片,再取一次計算total2,notidle2, cpuuse = (notidle2 – notidle) * 100 / (total2 - total)%
PS:由于Android 8權限收緊,在Android 8系統(tǒng)手機內apk內讀取文件內容為空,需要shell權限才可獲取文件內容,下同
讀/sys/devices/system/cpu/cpuX/cpufreq/scaling_cur_freq文件的值,X不定,看是幾核手機,scaling_cur_freq是否存在也不一定,需要判斷
至于為啥不取cpuinfo_cur_freq文件的值,原因是android 6,7系統(tǒng)獲取的時候,這個文件shell沒有讀取權限,需要root權限
參考文章:
Android 6,7系統(tǒng)可執(zhí)行
dumpsys window windows | grep "mCurrentFocus"
執(zhí)行結果一般為類似:
mCurrentFocus=Window{81caaa5 u0 com.tencent.mobileqq/com.tencent.mobileqq.activity.SplashActivity}
按照一定規(guī)則把com.tencent.mobileqq提取出來即可
直接apk內讀取文件即可,不需要shell權限(支持到Android8)
Gpu使用率獲取:會得到兩個值,(前一個/后一個)*100%=使用率
adb shell cat /sys/class/kgsl/kgsl-3d0/gpubusy
Gpu工作頻率:
adb shell cat /sys/class/kgsl/kgsl-3d0/gpuclk
adb shell cat /sys/class/kgsl/kgsl-3d0/devfreq/cur_freq
Gpu最大、最小工作頻率:
adb shell cat /sys/class/kgsl/kgsl-3d0/devfreq/max_freq
adb shell cat /sys/class/kgsl/kgsl-3d0/devfreq/min_freq
Gpu可用頻率
adb shell cat /sys/class/kgsl/kgsl-3d0/gpu_available_frequencies
adb shell cat /sys/class/kgsl/kgsl-3d0/devfreq/available_frequencies
Gpu可用工作模式:
adb shell cat /sys/class/kgsl/kgsl-3d0/devfreq/available_governors
Gpu當前工作模式:
adb shell cat /sys/class/kgsl/kgsl-3d0/devfreq/governor
直接按DEL+CTRL+ALT鍵,調出電腦任務管理器,查看性能,就可以看到有CPU占用率,核心數(shù)占用率的顯示。
或者安裝常見的魯大師軟件,在溫度欄下,也可以直接看到CPU的占用率。
獲取 CPU 核數(shù) Linux 中的設備都是以文件的形式存在,CPU 也不例外,因此 CPU 的文件個數(shù)就等價與核數(shù)。 Android 的 CPU 設備文件位于/sys/devices/system/cpu/目錄,文件名的的格式為cpu\d+。 root@generic_x86_64:/sys/devices/system/cpu # ls cpu0 cpufreq cpuidle kernel_max modalias offline online possible power present uevent統(tǒng)計一下文件個數(shù)便可以獲得 CPU 核數(shù)。 public static int getNumberOfCPUCores() {if (Build.VERSION.SDK_INT = Build.VERSION_CODES.GINGERBREAD_MR1) {// Gingerbread doesn't support giving a single application access to both cores, but a// handful of devices (Atrix 4G and Droid X2 for example) were released with a dual-core// chipset and Gingerbread; that can let an app in the background run without impacting// the foreground application. But for our purposes, it makes them single core.return 1;}int cores;try {cores = new File("/sys/devices/system/cpu/").listFiles(CPU_FILTER).length;} catch (SecurityException e) {cores = DEVICEINFO_UNKNOWN;} catch (NullPointerException e) {cores = DEVICEINFO_UNKNOWN;}return cores; }private static final FileFilter CPU_FILTER = new FileFilter() {@Overridepublic boolean accept(File pathname) {String path = pathname.getName();//regex is slow, so checking char by char.if (path.startsWith("cpu")) {for (int i = 3; i path.length(); i++) {if (path.charAt(i) '0' path.charAt(i) '9') {return false;}}return true;}return false;} };獲取時鐘頻率 獲取時鐘頻率需要讀取系統(tǒng)文件 -/sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq或者/proc/cpuinfo。Android 模擬器中并沒有cpuinfo_max_freq文件,因此只能讀取/proc/cpuinfo。 /proc/cpuinfo包含了很多 cpu 數(shù)據(jù)。 processor : 0 vendor_id : GenuineIntel cpu family : 6 model : 70 model name : Intel(R) Core(TM) i7-4770HQ CPU @ 2.20GHz stepping : 1 cpu MHz : 0.000 cache size : 1024 KB fdiv_bug : no hlt_bug : no f00f_bug : no coma_bug : no fpu : yes fpu_exception : yes cpuid level : 4 wp : yes代碼如下: public static int getCPUMaxFreqKHz() {int maxFreq = DEVICEINFO_UNKNOWN;try {for (int i = 0; i getNumberOfCPUCores(); i++) {String filename ="/sys/devices/system/cpu/cpu" + i + "/cpufreq/cpuinfo_max_freq";File cpuInfoMaxFreqFile = new File(filename);if (cpuInfoMaxFreqFile.exists()) {byte[] buffer = new byte[128];FileInputStream stream = new FileInputStream(cpuInfoMaxFreqFile);try {stream.read(buffer);int endIndex = 0;//Trim the first number out of the byte buffer.while (buffer[endIndex] = '0' buffer[endIndex] = '9' endIndex buffer.length) endIndex++;String str = new String(buffer, 0, endIndex);Integer freqBound = Integer.parseInt(str);if (freqBound maxFreq) maxFreq = freqBound;} catch (NumberFormatException e) {//Fall through and use /proc/cpuinfo.} finally {stream.close();}}}if (maxFreq == DEVICEINFO_UNKNOWN) {FileInputStream stream = new FileInputStream("/proc/cpuinfo");try {int freqBound = parseFileForValue("cpu MHz", stream);freqBound *= 1000; //MHz - kHzif (freqBound maxFreq) maxFreq = freqBound;} finally {stream.close();}}} catch (IOException e) {maxFreq = DEVICEINFO_UNKNOWN; //Fall through and return unknown.}return maxFreq; }獲取內存大小 如果 SDK 版本大于等于JELLY_BEAN,可以通過ActivityManager來獲取內從大小。 ActivityManager.MemoryInfo memInfo = new ActivityManager.MemoryInfo(); ActivityManager am = (ActivityManager) c.getSystemService(Context.ACTIVITY_SERVICE); am.getMemoryInfo(memInfo);如果版本低于JELLY_BEAN,則只能讀取系統(tǒng)文件了。 FileInputStream stream = new FileInputStream("/proc/meminfo"); totalMem = parseFileForValue("MemTotal", stream);完整代碼如下: @TargetApi(Build.VERSION_CODES.JELLY_BEAN) public static long getTotalMemory(Context c) {// memInfo.totalMem not supported in pre-Jelly Bean APIs.if (Build.VERSION.SDK_INT = Build.VERSION_CODES.JELLY_BEAN) {ActivityManager.MemoryInfo memInfo = new ActivityManager.MemoryInfo();ActivityManager am = (ActivityManager) c.getSystemService(Context.ACTIVITY_SERVICE);am.getMemoryInfo(memInfo);if (memInfo != null) {return memInfo.totalMem;} else {return DEVICEINFO_UNKNOWN;}} else {long totalMem = DEVICEINFO_UNKNOWN;try {FileInputStream stream = new FileInputStream("/proc/meminfo");try {totalMem = parseFileForValue("MemTotal", stream);totalMem *= 1024;} finally {stream.close();}} catch (IOException e) {}return totalMem;} }