Android中如何查看記憶體(上)

來源:互聯網
上載者:User

標籤:

文章參照自:http://stackoverflow.com/questions/2298208/how-to-discover-memory-usage-of-my-application-in-android#2299813像Linux這種現代作業系統的記憶體使用量是很複雜的,因此很難準確的知道你的應用程式使用了好多記憶體。查看記憶體使用量的方式有很多種,但是各個方式查看到的結果可能會有微略不同。方式一,Running services通過手機上Running services的Activity查看,可以通過Setting->Applications->Running services進。關於Running services的詳細內容請參考《 Android中使用"running services"查看service進程記憶體》方式二,使用ActivityManager的getMemoryInfo(ActivityManager.MemoryInfo outInfo)ActivityManager.getMemoryInfo()主要是用於得到當前系統剩餘記憶體的及判斷是否處於低記憶體運行。執行個體1:    private void displayBriefMemory() {            final ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);            ActivityManager.MemoryInfo info = new ActivityManager.MemoryInfo();           activityManager.getMemoryInfo(info);            Log.i(tag,"系統剩餘記憶體:"+(info.availMem >> 10)+"k");           Log.i(tag,"系統是否處於低記憶體運行:"+info.lowMemory);        Log.i(tag,"當系統剩餘記憶體低於"+info.threshold+"時就看成低記憶體運行");    } ActivityManager.getMemoryInfo()是用ActivityManager.MemoryInfo返回結果,而不是Debug.MemoryInfo,他們不一樣的。ActivityManager.MemoryInfo只有三個Field:availMem:表示系統剩餘記憶體lowMemory:它是boolean值,表示系統是否處於低記憶體運行hreshold:它表示當系統剩餘記憶體低於好多時就看成低記憶體運行方式三,在代碼中使用Debug的getMemoryInfo(Debug.MemoryInfo memoryInfo)或ActivityManager的MemoryInfo[] getProcessMemoryInfo(int[] pids)該方式得到的MemoryInfo所描述的記憶體使用量情況比較詳細.資料的單位是KB.MemoryInfo的Field如下dalvikPrivateDirty: The private dirty pages used by dalvik。dalvikPss :The proportional set size for dalvik.dalvikSharedDirty :The shared dirty pages used by dalvik.nativePrivateDirty :The private dirty pages used by the native heap.nativePss :The proportional set size for the native heap.nativeSharedDirty :The shared dirty pages used by the native heap.otherPrivateDirty :The private dirty pages used by everything else.otherPss :The proportional set size for everything else.otherSharedDirty :The shared dirty pages used by everything else.Android和Linux一樣有大量記憶體在進程之間進程共用。某個進程準確的使用好多記憶體實際上是很難統計的。因為有paging out to disk(換頁),所以如果你把所有映射到進程的記憶體相加,它可能大於你的記憶體的實際物理大小。dalvik:是指dalvik所使用的記憶體。native:是被native堆使用的記憶體。應該指使用C\C++在堆上分配的記憶體。other:是指除dalvik和native使用的記憶體。但是具體是指什麼呢?至少包括在C\C++分配的非堆記憶體,比如分配在棧上的記憶體。puzlle!private:是指私人的。非共用的。share:是指共用的記憶體。PSS:實際使用的實體記憶體(比例分配共用庫佔用的記憶體)Pss:它是把共用記憶體根據一定比例分攤到共用它的各個進程來計算所得到進程使用記憶體。網上又說是比例分配共用庫佔用的記憶體,那麼至於這裡的共用是否只是庫的共用,還是不清楚。 PrivateDirty:它是指非共用的,又不能換頁出去(can not be paged to disk )的記憶體的大小。比如Linux為了提高分配記憶體速度而緩衝的小對象,即使你的進程結束,該記憶體也不會釋放掉,它只是又重新回到緩衝中而已。 SharedDirty:參照PrivateDirty我認為它應該是指共用的,又不能換頁出去(can not be paged to disk )的記憶體的大小。比如Linux為了提高分配記憶體速度而緩衝的小對象,即使所有共用它的進程結束,該記憶體也不會釋放掉,它只是又重新回到緩衝中而已。具體代碼請參考執行個體1注意1:MemoryInfo所描述的記憶體使用量情況都可以通過命令adb shell "dumpsys meminfo %curProcessName%" 得到。注意2:如果想在代碼中同時得到多個進程的記憶體使用量或非本進程的記憶體使用量情況請使用ActivityManager的MemoryInfo[] getProcessMemoryInfo(int[] pids),否則Debug的getMemoryInfo(Debug.MemoryInfo memoryInfo)就可以了。注意3:可以通過ActivityManager的List<ActivityManager.RunningAppProcessInfo> getRunningAppProcesses()得到當前所有啟動並執行進程資訊。ActivityManager.RunningAppProcessInfo中就有進程的id,名字以及該進程包括的所有apk包名列表等。注意4:資料的單位是KB.方式4、使用Debug的getNativeHeapSize (),getNativeHeapAllocatedSize (),getNativeHeapFreeSize ()方法。該方式只能得到Native堆的記憶體大概情況,資料單位為位元組。static long getNativeHeapAllocatedSize() Returns the amount of allocated memory in the native heap.返回的是當前進程navtive堆中已使用的記憶體大小
static long getNativeHeapFreeSize()Returns the amount of free memory in the native heap.返回的是當前進程navtive堆中已經剩餘的記憶體大小static long getNativeHeapSize()Returns the size of the native heap.返回的是當前進程navtive堆本身總的記憶體大小範例程式碼:          Log.i(tag,"NativeHeapSizeTotal:"+(Debug.getNativeHeapSize()>>10));          Log.i(tag,"NativeAllocatedHeapSize:"+(Debug.getNativeHeapAllocatedSize()>>10));          Log.i(tag,"NativeAllocatedFree:"+(Debug.getNativeHeapFreeSize()>>10));注意:DEBUG中居然沒有與上面相對應的關於dalvik的函數。方式五、使用dumpsys meminfo命令。我們可以在adb shell 中運行dumpsys meminfo命令來得到進程的記憶體資訊。在該命令的後面要加上進程的名字,以確定是哪個進程。比如"adb shell dumpsys meminfo com.teleca.robin.test" 將得到com.teleca.robin.test進程使用的記憶體的資訊: Applications Memory Usage (kB):Uptime: 12101826 Realtime: 270857936 ** MEMINFO in pid 3407 [com.teleca.robin.test] **                    native   dalvik    other    total            size:     3456     3139      N/A     6595       allocated:     3432     2823      N/A     6255            free:       23      316      N/A      339           (Pss):      724     1101     1070     2895  (shared dirty):     1584     4540     1668     7792    (priv dirty):      644      608      688     1940  Objects           Views:        0        ViewRoots:        0     AppContexts:        0       Activities:        0          Assets:        3    AssetManagers:        3   Local Binders:        5    Proxy Binders:       11Death Recipients:        0 OpenSSL Sockets:        0  SQL            heap:        0       memoryUsed:        0pageCacheOverflo:        0  largestMemAlloc:        0   Asset Allocations    zip:/data/app/com.teleca.robin.test-1.apk:/resources.arsc: 1K "size" 表示的是總記憶體大小(kb)。, "allocated" 表示的是已使用了的記憶體大小(kb),, "free"表示的是剩餘的記憶體大小(kb), 更多的可以參照方式三和方式四中的描述現在已經有了自動提取匯總dumpsys meminfo資訊的工具,具體請參照《 Android記憶體泄露利器(記憶體統計篇)》及其系列文章。 方式六、使用 "adb shell procrank"命令如果你想查看所有進程的記憶體使用量情況,可以使用"adb shell procrank"命令。命令返回將如下:  PID      Vss      Rss      Pss      Uss  cmdline  188   75832K   51628K   24824K   19028K  system_server  308   50676K   26476K    9839K    6844K  system_server 2834   35896K   31892K    9201K    6740K  com.sec.android.app.twlauncher  265   28536K   28532K    7985K    5824K  com.android.phone  100   29052K   29048K    7299K    4984K  zygote  258   27128K   27124K    7067K    5248K  com.swype.android.inputmethod  270   25820K   25816K    6752K    5420K  com.android.kineto 1253   27004K   27000K    6489K    4880K  com.google.android.voicesearch 2898   26620K   26616K    6204K    3408K  com.google.android.apps.maps:FriendService  297   26180K   26176K    5886K    4548K  com.google.process.gapps 3157   24140K   24136K    5191K    4272K  android.process.acore 2854   23304K   23300K    4067K    2788K  com.android.vending 3604   22844K   22840K    4036K    3060K  com.wssyncmldm  592   23372K   23368K    3987K    2812K  com.google.android.googlequicksearchbox 3000   22768K   22764K    3844K    2724K  com.tmobile.selfhelp  101    8128K    8124K    3649K    2996K  /system/bin/mediaserver 3473   21792K   21784K    3103K    2164K  com.android.providers.calendar 3407   22092K   22088K    2982K    1980K  com.teleca.robin.test 2840   21380K   21376K    2953K    1996K  com.sec.android.app.controlpanel......................................................................................................................關於VSS,RSS,PSS,USS的意義請參考《 Android記憶體之VSS/RSS/PSS/USS》注意1:這裡的PSS和方式四PSS的total並不一致,有細微的差別。為什麼呢?這是因為procrank 命令和meminfo命令使用的核心機制不太一樣,所以結果會有細微差別注意2:這裡的Uss 和方式四的Priv Dirtyd的total幾乎相等.他們似乎表示的是同一個意義。但是現在得到的關於它們的意義的解釋卻不太相同。難道這裡Private的都是dirty(這裡指不能換頁)? Puzzle! 方式七、使用"adb shell cat /proc/meminfo" 命令。該方式只能得出系統整個記憶體的大概使用方式。MemTotal:         395144 kB MemFree:          184936 kB Buffers:             880 kB Cached:            84104 kB SwapCached:            0 kB ................................................................................................MemTotal :可供系統和使用者使用的總記憶體大小 (它比實際的實體記憶體要小,因為還有些記憶體要用於radio, DMA buffers, 等). MemFree:剩餘的可用記憶體大小。這裡該值比較大,實際上一般Android system 的該值通常都很小,因為我們盡量讓進程都保持運行,這樣會耗掉大量記憶體。Cached: 這個是系統用於檔案緩衝等的記憶體. 通常systems需要20MB 以避免bad paging states;。當記憶體緊張時,the Android out of memory killer將殺死一些background進程,以避免他們消耗過多的cached RAM ,當然如果下次再用到他們,就需要paging. 那麼是說background進程的記憶體包含在該項中嗎? 方式八,使用“adb shell ps -x”命令該方式主要得到的是記憶體資訊是VSIZE 和RSS。USER     PID   PPID  VSIZE  RSS     WCHAN    PC         NAME.........................省略.................................app_70    3407  100   267104 22056 ffffffff afd0eb18 S com.teleca.robin.test (u:55, s:12)app_7     3473  100   268780 21784 ffffffff afd0eb18 S com.android.providers.calendar (u:16, s:8)radio     3487  100   267980 21140 ffffffff afd0eb18 S com.osp.app.signin (u:11, s:12)system    3511  100   273232 22024 ffffffff afd0eb18 S com.android.settings (u:11, s:4)app_15    3546  100   267900 20300 ffffffff afd0eb18 S com.sec.android.providers.drm (u:15, s:6)app_59    3604  100   272028 22856 ffffffff afd0eb18 S com.wssyncmldm (u:231, s:54)root      4528  2     0      0     c0141e4c 00000000 S flush-138:13 (u:0, s:0)root      4701  152   676    336   c00a68c8 afd0e7cc S /system/bin/sh (u:0, s:0)root      4702  4701  820    340   00000000 afd0d8bc R ps (u:0, s:5)VSZIE:意義暫時不明。VSS:請參考《 Android記憶體之VSS/RSS/PSS/USS》注意1:由於RSS的價值不是很大,所以一般不用。注意2:通過該命令提取RSS,已經有了工具,具體參照《 Android記憶體泄露利器(RSS記憶體統計篇)》及其系列。

Android中如何查看記憶體(上)

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.