一世 我们知道。android程序存储器通常被限制16M。当然,24M的,和android程序存储器分为2部分:native和dalvik。dalvik 就是我们寻常说的java堆。我们创建的对象是在这里面分配的,而bitmap是直接在native上分配的,对于内存的限制是 native+dalvik 不能超过最大限制。

1.单个app 内存大小限制

[java] view plaincopy
  1. ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
  2. activityManager.getMemoryClass();

2."/proc/meminfo"  系统内存信息文件【整个系统的内存情况】

Android获取手机总内存和可用内存

“/proc/meminfo”解析

3.ActivityManager.MemoryInfo【整个系统】

availMem         剩余内存

threshold         临界值 【超过次值就開始杀死后台服务和没有关联的进程】

lowMemory     低内存状态

4.android.os.Debug

getNativeHeapFreeSize()

getNativeHeapAllocatedSize()

getNativeHeapSize()

5.android.os.Debug.MemoryInfo【当前进程的内存情况】


6.Runtime


totalMemory()
freeMemory()

maxMemory()

7.VMRuntime

getTargetHeapUtilization()

getMinimumHeapSize()

getExternalBytesAllocated()   应该是外部分配的内存Native内存

8.GC_EXTERNAL_ALLOC freed 与 GC_EXPLICIT freed 是什么?

系统GC释放的内存提示

一般在LOG里面显演示样例如以下:
09-28 17:16:37.543: DEBUG/dalvikvm(21466): GC_EXTERNAL_ALLOC freed 390 objects / 45656 bytes in 50ms
09-28 17:16:40.513: DEBUG/dalvikvm(3267): GC_EXPLICIT freed 4501 objects / 251624 bytes in 67ms

EXPLICIT:Free的内存是VM中java使用的内存 即 heap mem

EXTERNA:Free的内存是VM中通过JNI的Native类中的malloc分配的内存 比如 Bitmap 和一些 Cursor

在Davilk中,给一个程序分配的内存依据机型厂商的不同,而不同。如今的大部分的是32M了。而在VM内部会把这些内存分成java使用的内存和 Native使用的内存,它们之间是不能共享的。就是说当你的Native内存用完了。如今Java又有空暇的内存。这时Native会又一次像VM申请,而不是直接使用java的。

比如上边的样例

explicit 3411K/6663K

external 24870K/26260K

假设这时须要创建一个2M的

Bitmap,

Native现有内存26260-24870=1390K<2048k,因此他就会向Vm申请内存,尽管java空暇的内存是

6663-3411=3252>2048,但这部分内存Native是不能使用。

可是你如今去申请2M的Native内存,VM会告诉你无法分配的,由于如今已使用的内存已经接近峰值了32M(26260+6663=32923 ),所以如今就会成force close 报OOM。

所以如今我们要检查我们的native内存的使用情况来避免OOM。

ps: http://stackoverflow.com/questions/2298208/how-to-discover-memory-usage-of-my-application-in-android#2299813

三、通过Android系统提供的Runtime类,运行adb 命令(top,procrank,ps...等命令)查询

内存耗用:VSS/RSS/PSS/USS
Terms
? VSS - Virtual Set Size 虚拟耗用内存(包括共享库占用的内存)
?

RSS - Resident Set Size 实际使用物理内存(包括共享库占用的内存)
? PSS - Proportional Set Size 实际使用的物理内存(比例分配共享库占用的内存)
? USS - Unique Set Size 进程独自占用的物理内存(不包括共享库占用的内存)
一般来说内存占用大小有例如以下规律:VSS >= RSS >= PSS >= USS

查看每一个进程及其内存状况

 private void getRunningAppProcessInfo() {mActivityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);// 获得系统里正在执行的全部进程List<RunningAppProcessInfo> runningAppProcessesList = mActivityManager.getRunningAppProcesses();for (RunningAppProcessInfo runningAppProcessInfo : runningAppProcessesList) {// 进程ID号int pid = runningAppProcessInfo.pid;// 用户IDint uid = runningAppProcessInfo.uid;// 进程名String processName = runningAppProcessInfo.processName;// 占用的内存int[] pids = new int[] { pid };Debug.MemoryInfo[] memoryInfo = mActivityManager.getProcessMemoryInfo(pids);int memorySize = memoryInfo[0].dalvikPrivateDirty;st = st + "processName=" + processName + ",pid=" + pid + ",uid="+ uid + ",memorySize=" + memorySize + "kb" + "\n";System.out.println("processName=" + processName + ",pid=" + pid+ ",uid=" + uid + ",memorySize=" + memorySize + "kb");}}

查看总内存:

  public long getmem_TOLAL() {long mTotal;// /proc/meminfo读出的内核信息进行解释String path = "/proc/meminfo";String content = null;BufferedReader br = null;try {br = new BufferedReader(new FileReader(path), 8);String line;if ((line = br.readLine()) != null) {content = line;}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally {if (br != null) {try {br.close();} catch (IOException e) {e.printStackTrace();}}}// beginIndexint begin = content.indexOf(':');// endIndexint end = content.indexOf('k');// 截取字符串信息content = content.substring(begin + 1, end).trim();mTotal = Integer.parseInt(content);return mTotal;}

查看内存信息(该api较新):

   public long getmem_UNUSED(Context mContext) {long MEM_UNUSED;// 得到ActivityManagerActivityManager am = (ActivityManager) mContext.getSystemService(Context.ACTIVITY_SERVICE);// 创建ActivityManager.MemoryInfo对象ActivityManager.MemoryInfo mi = new ActivityManager.MemoryInfo();am.getMemoryInfo(mi);textView3.setText("totalMen:" + mi.totalMem / 1024 + "\n"+ "threshold:" + mi.threshold / 1024 + "\n" + "availMem:"+ mi.availMem / 1024 + "\n");// 取得剩余的内存空间MEM_UNUSED = mi.availMem / 1024;return MEM_UNUSED;}

查看app内存:

    ActivityManager manager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);int i=manager.getMemoryClass();textView.setText("\n"+"app:"+i);
像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<<a href="http://developer.android.com/reference/android/app/ActivityManager.RunningAppProcessInfo.html" rel="nofollow" style="color: rgb(207, 121, 28); text-decoration: none;">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:       11
Death Recipients:        0
OpenSSL Sockets:        0
SQL
heap:        0       memoryUsed:        0
pageCacheOverflo:        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内存统计篇)》及其系列。

实例1

int cnt=0;

final static int kBufferMinSize=1000;

final static int kBufferMaxSize=2000;

StringBuffer strBuffer=new StringBuffer(kBufferMinSize);

StringBuffer strBuffer2=new StringBuffer(kBufferMinSize);

StringBuffer strBuffer3=new StringBuffer(kBufferMinSize);

StringBuffer strBufferNativePss=new StringBuffer(kBufferMinSize);

StringBuffer strBufferDalvikPss=new StringBuffer(kBufferMinSize);

StringBuffer strBufferOtherPss=new StringBuffer(kBufferMinSize);

Debug.MemoryInfo memoryInfo=new Debug.MemoryInfo();

final static String tag="robin";

void printMemory()

{

long totalMemory=Runtime.getRuntime().totalMemory();

long freeMemory=Runtime.getRuntime().freeMemory();

long usedMemory=(totalMemory-freeMemory)>>10;

totalMemory=totalMemory>>10;

freeMemory=freeMemory>>10;

if(strBuffer.length()>kBufferMaxSize)

{

strBuffer.delete(0,strBuffer.length());

strBuffer2.delete(0,strBuffer2.length());

strBuffer3.delete(0,strBuffer3.length());

strBufferNativePss.delete(0,strBufferNativePss.length());

strBufferDalvikPss.delete(0,strBufferDalvikPss.length());

}

strBuffer.append(usedMemory+",");

strBuffer2.append(totalMemory+",");

strBuffer3.append((Debug.getNativeHeapSize()>>10)+",");

Debug.getMemoryInfo(memoryInfo);

strBufferNativePss.append((memoryInfo.nativePss)+",");

strBufferDalvikPss.append((memoryInfo.dalvikPss)+",");

if(cnt++==0)

{

Log.i(tag,"usedMemory:"+strBuffer.toString());

Log.i(tag,"totalMemory:"+strBuffer2.toString());

Log.i(tag,"NativeHeapSize:"+strBuffer3.toString());

Log.i(tag,"Native PSS:"+strBufferNativePss.toString());

Log.i(tag,"Dalvik PSS:"+strBufferDalvikPss.toString());

}

}

注意。对于输出的内存信息日志,我们稍作编辑就能够用于在excel产中图表。比便更直观的进行分析。
http://blog.csdn.net/hudashi/article/details/7050897
http://blog.csdn.net/hudashi/article/details/7050905

版权声明:本文博主原创文章,博客,未经同意不得转载。

【Android先进】查看手机记忆库状态和应用方法相关推荐

  1. android service 休眠,保持Android Service在手机休眠后继续运行的方法

    最近项目中用到了service进行计时,在连接USB的情况下一切正常,但是拔掉USB后发现,手机进入休眠后service停止了工作.最后通过 PowerManager.WakeLock 在屏幕休眠后保 ...

  2. android获取录音读写权限设置,Android编程检测手机录音权限是否打开的方法

    本文实例讲述了Android编程检测手机录音权限是否打开的方法.分享给大家供大家参考,具体如下: 6.0之前的权限检测只是检测到是否在清单文件中注册 Boolean flag = (PackageMa ...

  3. android 录音权限 代码打开,Android编程检测手机录音权限是否打开的方法

    本文实例讲述了Android编程检测手机录音权限是否打开的方法.分享给大家供大家参考,具体如下: 6.0之前的权限检测只是检测到是否在清单文件中注册 Boolean flag = (PackageMa ...

  4. 如何查看手机型号,最好的查询方法

    在琢石模拟器的网站上,有一个非常实用而且强大的系统信息查询工具,能够帮我们查看手机的完整参数,让我们鉴别手机真伪跟判断手机参数是不是跟说明书一致. 1.从 网站页面下面附件地址进入百度网盘中. 2.下 ...

  5. 【Android高级】查看手机及应用内存状况的方法

    我 们知道,android程序内存一般限制在16M,当然也有24M的,而android程序内存被分为2部分:native和dalvik,dalvik 就是我们平常说的java堆,我们创建的对象是在这里 ...

  6. Android怎么查看手机中的本地数据库

    我前几天做的项目中有本地数据库, 所以就 用的 SQLite,在调试数据库时,,很想看一下里面的表结构是否正确,这个时候就十分苦恼, 因为这个db文件不能够直接拿出来,我们知道,在DDMS里面有一个F ...

  7. Android 监听手机GPS打开状态

    转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/70854942 本文出自[赵彦军的博客] GPS_Presenter package ...

  8. android adb查看手机内存使用情况

    1.查看设备节点内存adb shell df 2.查看磁盘内存和运行内存:adb shell free 3.查看内存详细数据adb shell cat /proc/meminfo MemTotal: ...

  9. Android studio 查看手机打印日志

    Android studio 配置SDK Android studio 配置adb 点击  View -Tool Windows - Logcat 打开日志

最新文章

  1. 使用inetaddress测试目标可达性_白盒测试工具―Winams介绍
  2. HD 1003 Max Sum (最大字段和问题)
  3. 热点的ap频段哪个快_WLAN中无线AP信道的划分
  4. SpringBoot整合Redis 主从复制_01
  5. 大数据之Elasticsearch教程
  6. 网络图片 base64 java_java图片转base64和真实的结果不一样
  7. i386 Linux 系统调用
  8. Who is in me? -- Freeware on Windows
  9. BUUCTF刷题记录(持续更新中~)
  10. android 英语单词音标获取
  11. 硬件nat关闭还是开启_卡顿未必怪硬件,Win10玩游戏不可不知的技巧
  12. 出轨的是老公为什么要老婆做出改变
  13. iOS 5G网络判断
  14. OracleP6机场工程进度控制系列15:总进度综合管控报告
  15. 2.04 标志寄存器
  16. 广州市越秀区2021-2022学年九年级第一学期期末考试英语试题
  17. 万年历c语言攀枝花,C语言实现万年历
  18. OpenLayers加载谷歌地球离线瓦片地图
  19. GOM引擎传奇中增加会员时间的脚本教程分享
  20. 使用Sublime Text+SumatraPDF轻松写论文

热门文章

  1. linux安装mysql5.6.14_(四)Zabbix_linux安装mysql5.6
  2. 用c语言编辑一个通讯录,C语言实现一个通讯录
  3. 模糊神经网络_神经网络模型:当网络开始产生类似于人类思维的过程
  4. html5 white space,CSS white-space 属性
  5. js如何调用h5的日期控价_微信公众号支付H5调用支付解析
  6. 如何用计算机组添加打印机共享的打印机,工作组内打印机如何共享?
  7. linux如何判断网线插入_斜口钳和网线钳制作网线!
  8. oracle01507报错,oracle报错ORA-01507
  9. Struts 1基础入门
  10. 判断一个int数组中的元素是否存在重复