当程序执行垃圾回收的时候,会打印一条Log信息。其格式例如以下:
D/dalvikvm: <GC_Reason> <Amount_freed>, <Heap_stats>, <External_memory_stats>, <Pause_time>
GC_Reason表示导致垃圾回收的原因以及当前的回收类型,包含下面几类:
GC_CONCURRENT:当堆中对象数量达到一定是触发的垃圾收集
GC_FOR_MALLOC:在内存已满的情况下分配内存,此时系统会暂停程序并回收内存
GC_HPROF_DUMP_HEAP:创建FPFOR文件来分析Heap时所造成的垃圾收集
GC_EXPLICIT: 程序调用了垃圾收集函数System.gc
GC_EXTERNAL_ALLOC: 出如今API 10及下面。为外部分配内存(native memory or NIO buffer)所造成的垃圾回收,高版本号所有分配在Dalvik Heap中。
Amount_freed 表示此次回收的内存
Heap_stats 表示空暇内存百分比和存活对象大小/堆的总大小
External_memory_stats 表示API 10及下面的外部分配内存。已分配内存/导致垃圾回收的界限
Pause_time 暂停时间。一个表示開始回收垃圾的时间。还有一个表示回收结束的暂停时间
D/dalvikvm( 9050): GC_CONCURRENT freed 2049K, 65% free 3571K/9991K, external 4703K/5261K, paused 2ms+2ms
注意这条信息中的 “ 3571K/9991K” 值,这代表着程序使用的heap大小。
2.使用DDMS
Eclipse中的DDMS提供了一个观察内存使用情况的GUI,当我们不断点击Cause GC时。就会看到当前程序的Heap,使用比較方便,详细使用方法能够Google一下。
3.使用adb dumpsys 命令
adb是一个很强大的工具,使用adb查看应用程序内存使用情况可按例如以下格式在命令行里查看内存使用情况:
adb shell dumpsys meminfo <package_name>
当中,package_name 也能够换成程序的pid。pid能够通过 adb shell top | grep app_name 来查找,下图是某个程序的内存使用情况:
重点关注例如以下几个字段:
(1) Native/Dalvik 的 Heap 信息
详细在上面的第一行和第二行,它分别给出的是JNI层和Java层的内存分配情况,假设发现这个值一直增长,则代表程序可能出现了内存泄漏。
(2) Total 的 PSS 信息
这个值就是你的应用真正占领的内存大小,通过这个信息。你能够轻松判别手机中哪些程序占内存比較大了。
4. 使用adb shell procrank
手机中的sh是经过精简过的。有些手机可能没有 procrank 命令。能够使用genymotion模拟器。或是自己安装procrank命令。使用procrank时,命令行的输出入下图:
能够看到,在linux下表示内存的耗用情况有四种不同的表现形式:
VSS - Virtual Set Size 虚拟耗用内存(包括共享库占用的内存)
RSS - Resident Set Size 实际使用物理内存(包括共享库占用的内存)
PSS - Proportional Set Size 实际使用的物理内存(比例分配共享库占用的内存)
USS - Unique Set Size 进程独自占用的物理内存(不包括共享库占用的内存)
VSS:VSS表示一个进程可訪问的所有内存地址空间的大小。
这个大小包含了进程已经申请但尚未使用的内存空间。在实际中非常少用这样的方式来表示进程占用内存的情况,用它来表示单个进程的内存使用情况是不准确的。
RSS:表示一个进程在RAM中实际使用的空间地址大小。包含了所有共享库占用的内存。这样的表示进程占用内存的情况也是不准确的。
PSS:表示一个进程在RAM中实际使用的空间地址大小,它按比例包括了共享库占用的内存。假如有3个进程使用同一个共享库,那么每一个进程的PSS就包括了1/3大小的共享库内存。
这样的方式表示进程的内存使用情况较准确。但当仅仅有一个进程使用共享库时,其情况和RSS一模一样。
USS:表示一个进程本身占用的内存空间大小,不包括其他不论什么成分,这是表示进程内存大小的最好方式!
能够看到:VSS>=RSS>=PSS>=USS
5.其他经常使用命令命令:
adb shell kill PIDNumber 死你想杀死的后台进程来模拟某种 bug 的复现条件。
adb shell ps 查看当前终端中的进程信息
那么怎样在代码中推断当前的硬件系统有多少的 RAM 呢?在 Framework ProcessList.java 中有例如以下代码可用:
ProcessList() {
MemInfoReader minfo = new MemInfoReader();
minfo.readMemInfo();
mTotalMemMb = minfo.getTotalSize()/(1024*1024);
}
查看进程占用cpu的情况:adb shell top -n 1 -d 0.5 | grep proc_ id