1.jvm 堆内存(dalvik 堆内存)

  • 每个Java应用程序在运行时都会拥有自己的JVM实例,这个实例会为其分配独立的堆内存空间。这意味着不同的应用程序之间不会共享堆内存。

不同手机中app进程的 jvm 堆内存是不同的,因厂商在出厂设备时会自定义设置其峰值。比如,在Android Studio 创建模拟器时,会设置jvm heap 默认384m , 如下图所示:

当app 进程中java 层 new 对象(加起来总和)占用的堆内存达到jvm heap 峰值时,就会抛出OOM 。

当app 进程中java 层 new 对象(加起来总和)占用的堆内存达到jvm heap 峰值时,就会抛出OOM 。

通过一个案例进一步,了解jvm 堆内存

通过以下代码,可获取到进程中jvm 堆内存的使用情况:

    public JSONObject statisticsJVMMemory() {
JSONObject json = new JSONObject();
Runtime runtime = Runtime.getRuntime();
//进程中最大jvm 内存大小
long max_memory = runtime.maxMemory() / 1024;
//进程中申请的jvm内存大小,不等用于一定分配那么多内存(会随着时间变化而变化)
long apply_memory = runtime.totalMemory() / 1024;
//进程中申请内存中可使用的jvm内存大小
long free_memory = runtime.freeMemory() / 1024;
//进程中已经使用的jvm 内存
long use_memory = apply_memory - free_memory;
//计算出jvm 的内存使用率,超过0.8就需要警惕,可能java 层内存存在泄漏
float use_memory_rate = ((float) use_memory) / max_memory;
//....
}

先来了解jvm 堆内存的几个指标:

1.最大限制内存: maxMemory,出厂时设置的

2.申请的内存: totalMemory,不等用于一定分配那么多内存(会随着时间变化而变化)

3.(申请的内存中)剩余使用的内存:freeMemory

4.已使用的内存: use_memory=totalMemory -freeMemory, 重点关注是这个

5.内存使用率: use_memory/maxMemory

模拟jvm 堆内存一直上涨的场景, 启动一个线程,周期性间隔几秒,不断模拟创建byte 数组, 然后统计app 进程的jvm 堆内存使用情况 :

        private List<byte[]> jvmLeakList = new ArrayList<>();
public void addJvmLeak() {
byte[] largeByte = new byte[50 * 1024 * 1024];
for (int i = 0; i < largeByte.length; ++i) {
largeByte[i] = 'a';//分配使用时,进程内存中物理内存才会使用
}
Log.i(TAG, "byte size: " + (largeByte.length / 1024 / 1024) + " mB");
jvmLeakList.add(largeByte);
}

查看输出日志 , 对比前后两次的堆内存变化:

多执行几次后,会触发oom 。先来看下oom 前的内存状况:

系统会主动触发gc ,输出art: Starting a blocking GC Alloc 和art: Alloc sticky concurrent mark sweep GC freed 0(0B) AllocSpace objects日志 ,解读如下:

当内存不足32m时,再次new 一个50M的byte 数组,就会抛出oom:

app 进程中真正剩余可用 jvm 堆内存是

       //真正可用的内存,包含剩余可申请的内存
long actual_free_memory=max_memory-use_memory;

处理jvm 内存不足的情况

当大型app或者游戏app 遇到 jvm 内存(大多数为384m)不足时,可通过android:largeHeap="true"来增加,jvm 堆内存 会调整为512M峰值。

2.native内存

获取app 进程的native 堆内存,代码如下:

    /**
* adb shell dumpsys meminfo packageName包名:
* 查看每个进程中内存状况(包含jvm 和native 、shareLib)
*
* @return
*/
private JSONObject statisticsNativeMemory() {
JSONObject jsonObject = new JSONObject();
// 当前进程中native层申请的堆内存,会随着时间而变化,加大或者减少
long totalNative = Debug.getNativeHeapSize() / 1024 / 1024;
//当进程中native层中已使用堆内存
long useNative = Debug.getNativeHeapAllocatedSize() / 1024 / 1024;
//当前进程中native层中剩余的堆内存
long freeNative = Debug.getNativeHeapFreeSize() / 1024 / 1024;
//....
}

在android 8.0 以后,bitamp 所占内存从jvm 堆内存移到native内存中,大大减少了oom的风险:

通过一个案例来进一步了解:

在android 8.0以上非64位的设备验证:

      public void addNativeLeak() {
/**
* 在android 8.0 以上版本, bitmap内存放到native层内存中,非jvm 堆内存中
*
* Bitmap.Config.ARGB_8888:一个像素占用4 byte
* Bitmap的内存大小=像素*4byte
*/
int width = 1024 * 8;
int height = 1024 * 8;
//bitmap 被创建时,会申请虚拟内存 256m=1024*8*1024*8*4
Bitmap bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
boolean draw=false;
if (draw) {
//当给bitmap绘制内容时,物理内存会增加; 物理内存是当真正需要使用时才会用到
Canvas canvas = new Canvas(bitmap);
paint.setAntiAlias(true);
paint.setColor(Color.WHITE);
canvas.drawCircle(width / 2, height / 2, width / 2, paint);
}
int pictureSize = bitmap.getByteCount() / 1024 / 1024;
int bitmapSize = bitmap.getAllocationByteCount() / 1024 / 1024;
Log.i(TAG, "bitmap size: " + bitmapSize + " mB" + " ,picture size: " + pictureSize + " mB");
nativeLeakList.add(bitmap);
}

启动一个线程,间隔几秒, 创建一个256m的bitmap ,输出当前native 内存情况:

当native 堆内存不断上涨,虚拟内存也会增加,直到oom。系统会输出Starting a blocking GC NativeAlloc标识当前native层内存不足。来看下, oom 前的native 内存情况:

3.app 进程中虚拟内存

先来了解下进程中虚拟内存与物理内存 概念

PSS(Proportional Set Size): 物理内存,PSS = USS + 按比例包含共享库

RSS(Resident Set Size): 物理内存,RSS = USS + 包含共享库

VSS(Virtual Set Size): 虚拟内存,VSS = RSS + 未分配实际物理内存

虚拟内存: 虚拟内存是计算机系统内存管理的一种技术。它使得应用程序认为它拥有连续的可用的内存(一个连续完整的地址空间),而实际上,它通常是被分隔成多个物理内存碎片,还有部分暂时存储在外部磁盘存储器上,在需要时进行数据交换(来源百度百科)。在32位的app进程中,虚拟内存最大4G,往往3G就oom了。

更多虚拟内存知识,请阅读虚拟内存-维基百科。

获取app进程中虚拟内存和物理内存的方式

    /**
* 计算进程中内存状况和线程状况:
* FDSize: 128 // 当前分配的文件描述符,这个值不是当前进程使用文件描述符的上线
* VmPeak: 4403108 kB // 当前进程运行过程中所占用内存的峰值
* VmSize: 4402056 kB // 已用逻辑空间地址,虚拟内存大小。整个进程使用虚拟内存大小,是VmLib, VmExe, VmData, 和 VmStk的总和。
* VmLck: 0 kB
* VmPin: 0 kB
* VmHWM: 49108 kB // 程序得到分配到物理内存的峰值
* VmRSS: 48920 kB // 程序现在正在使用的物理内存
* RssAnon: 9268 kB
* RssFile: 39540 kB
* RssShmem: 112 kB
* VmData: 1737808 kB // 所占用的虚拟内存
* VmStk: 8192 kB // 任务在用户态的栈的大小 (stack_vm)
* VmExe: 20 kB // 程序所拥有的可执行虚拟内存的大小,代码段,不包括任务使用的库 (end_code-start_code)
* VmLib: 163804 kB // 被映像到任务的虚拟内存空间的库的大小 (exec_lib)
* VmPTE: 1000 kB // 该进程的所有页表的大小,单位:kb
* Threads: 17 // 当前的线程数
*
* @return
*/
public JSONObject statisticsProcessMemory() { JSONObject json = new JSONObject();
// Linux 的/proc/self/status文件。这个并不是一个真实存在的文件,而为 Linux 的一个内核接口
File file = new File(ProcCmd.cmd_app_status);
readFileLine(file, (line) -> {
try {
String s = null;
if (line.startsWith(MemoryKeys.ProcessKeys.key_threads)) {
//进程中线程的数量
s = line;
} else if (line.startsWith(MemoryKeys.ProcessKeys.key_vm_size)) {
//整个进程中虚拟内存的总和(= VmLib+VmExe+VmData+VmStk),会动态变化增加
s = line;
} else if (line.startsWith(MemoryKeys.ProcessKeys.key_vm_rss)) {
// 进程中当前物理内存,即系统实际在物理内存上分配给程序的内存
s = line;
}
//.....
});
}

还是以上面的bitmap 为例子,间隔几秒创建bitmap 时,看虚拟内存和物理内存的变化:

32位进程虚拟内存3G多的问题

等待多执行几次后,虚拟内存就耗尽,会oom:

在32位设备上app 进程中虚拟内存是3G 多,在一些沙盒插件化32位运行环境下,游戏项目很容易虚拟内存耗尽。

32位进程和64进程的内存分配情况如下所示

每个进程中虚拟内存都是隔离的,互不干扰。

4.手机系统内存(处理器内存)

每个手机的处理器内存都是出厂时设置的,处理器内存也是物理内存。

Android 内存分析(java native heap内存、虚拟内存、处理器内存.的更多相关文章

  1. 【Android端 APP 内存分析】使用工具进行APP的内存分析

    Android端可以通过adb 命令直接获取内存信息,当然Android studio也提供了对内存的监控分析工具,并且后续可以结合MAT做分析 今天介绍的是通过Android studio和MAT工 ...

  2. Android内存分析工具DDMS heap + MAT 安装和使用

    一  Java内存分析工具扫盲 如果像我一样一点都不了解,可以先进行内存分析工具扫盲   MAT介绍:     Eclipse Memory Analyzer(MAT)一个功能丰富的 JAVA 堆转储 ...

  3. java dump 内存分析 elasticsearch Bulk异常引发的Elasticsearch内存泄漏

    Bulk异常引发的Elasticsearch内存泄漏 2018年8月24日更新: 今天放出的6.4版修复了这个问题. 前天公司度假部门一个线上ElasticSearch集群发出报警,有Data Nod ...

  4. Android 怎样在java/native层改动一个文件的权限(mode)与用户(owner)?

    前言          欢迎大家我分享和推荐好用的代码段~~ 声明          欢迎转载.但请保留文章原始出处:          CSDN:http://www.csdn.net        ...

  5. android 如何分析java.lang.IllegalArgumentException: Cannot draw recycled bitmaps异常

    这类问题的分析,通常你需要找到bitmap对象已经在那个位置recyle,然后检查代码. 如何定位的位置,其中代码具有对bitmap 目的recyle.能够 Bitmap.java的recycle方法 ...

  6. java基础(一)~~内存分析

    1.java内存分析 2.2.面向对象内存分析 Java虚拟机的内存可以分为三个区域:栈(stack).堆(heap).方法区(method area): 2.2.1.栈 栈的特点如下: 1.栈描述的 ...

  7. Android 内存监测工具 DDMS --> Heap(转)

    DDMS 的全称是Dalvik Debug Monitor Service,它为我们提供例如:为测试设备截屏,针对特定的进程查看正在运行的线程以及堆信息.Logcat.广播状态信息.模拟电话呼叫.接收 ...

  8. Java数组之三种初始化及内存分析

    内存分析 Java内存 堆: 1.存放new的对象和数组 2.可以被所有的线程共享,不会存放别的对象引用 栈: 1.存放基本变量类型(会包含这个基本类型的具体数值) 2.引用对象的变量(会存放这个引用 ...

  9. Java实例化对象过程中的内存分配

    Java实例化对象过程中的内存分配: https://blog.csdn.net/qq_36934826/article/details/82685791 问题引入这里先定义一个很不标准的“书”类,这 ...

  10. Day008 三种初始化及内存分析

    三种初始化和内存分析 Java内存分析: 堆: 存放new的对象和数组. 可以被所有的线程共享,不会存放别的对象引用. 栈: 存放基本变量类型(会包含这个基本类型的具体数值). 引用对象的变量(会存放 ...

随机推荐

  1. 将txt转化为csv的方法和遇到问题

    一.无法修改扩展名步骤如下 二.转换之后所有数据都挤在第一列 win10系统修改文件扩展名只需4部,打开我的电脑->查看->选择->查看->取消勾选(已知隐藏文件的扩展名)-& ...

  2. 【WSDL】02 四种客户端调用方式

    WSDL概念和一些语法内容: https://www.w3school.com.cn/wsdl/index.asp SOAP概念: https://www.runoob.com/soap/soap-t ...

  3. 计算机领域:学术写作中的conducive的含义表示

    "Conducive" 的意思是"有助于"或"有益于".在学术和正式的写作中,"conducive" 常用于描述某种情况 ...

  4. 大模型时代该用什么样的显卡 —— 实验室新进两块A800显卡

    具体如图: (这两个显卡是专为实验室的大模型方向提供的) 关于A800显卡的性能参数: (上图源自:https://www.zhihu.com/question/618932114/answer/32 ...

  5. 树莓派3b+ 安装windows10 arm版本的方法及使用体验

    首先,我再网上找到了一个很详细的为树莓派3b安装windows10 arm的教程,实际操作下来发现并不可行. 最后找到了可行的教程: 第3章 将Windows10镜像写入TF卡:https://zhu ...

  6. Deepin20系统开机报错——You are in emergency mode ... Cannot open access to console, the root account is locked. emergency mode/“journalctl -xb”

    参考: https://knowledge.ipason.com/ipKnowledge/knowledgedetail.html/1286 https://blog.csdn.net/wenfei1 ...

  7. vue&element项目实战 之element使用&用户&字典模块实现

    6.用户模块 用户模块api import request from '@/utils/request' export function login(data) { return request({ ...

  8. Sentry For Vue 完整接入详解(2021 Sentry v21.8.x)前方高能预警!三万字,慎入!

    内容源于:https://docs.sentry.io/platforms/javascript/guides/vue/ 系列 1 分钟快速使用 Docker 上手最新版 Sentry-CLI - 创 ...

  9. 结构开发笔记(三):solidworks软件(二):小试牛刀,绘制一个立方体

    前言   solidworks草图大师,基本的使用过程.  所有的零件基础都是从平面绘制开始,然后凸出来厚度.  本篇绘制一个简单的立方体,熟悉基本操作.   立方体绘制过程   选取一个平面绘制一个 ...

  10. bat 随笔

    bat 获取文件名 %%~nxi bat 变量去除空字符 BAT批处理中的字符串处理详解(字符串截取)