Android 内存分析(java native heap内存、虚拟内存、处理器内存.
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内存、虚拟内存、处理器内存.的更多相关文章
- 【Android端 APP 内存分析】使用工具进行APP的内存分析
Android端可以通过adb 命令直接获取内存信息,当然Android studio也提供了对内存的监控分析工具,并且后续可以结合MAT做分析 今天介绍的是通过Android studio和MAT工 ...
- Android内存分析工具DDMS heap + MAT 安装和使用
一 Java内存分析工具扫盲 如果像我一样一点都不了解,可以先进行内存分析工具扫盲 MAT介绍: Eclipse Memory Analyzer(MAT)一个功能丰富的 JAVA 堆转储 ...
- java dump 内存分析 elasticsearch Bulk异常引发的Elasticsearch内存泄漏
Bulk异常引发的Elasticsearch内存泄漏 2018年8月24日更新: 今天放出的6.4版修复了这个问题. 前天公司度假部门一个线上ElasticSearch集群发出报警,有Data Nod ...
- Android 怎样在java/native层改动一个文件的权限(mode)与用户(owner)?
前言 欢迎大家我分享和推荐好用的代码段~~ 声明 欢迎转载.但请保留文章原始出处: CSDN:http://www.csdn.net ...
- android 如何分析java.lang.IllegalArgumentException: Cannot draw recycled bitmaps异常
这类问题的分析,通常你需要找到bitmap对象已经在那个位置recyle,然后检查代码. 如何定位的位置,其中代码具有对bitmap 目的recyle.能够 Bitmap.java的recycle方法 ...
- java基础(一)~~内存分析
1.java内存分析 2.2.面向对象内存分析 Java虚拟机的内存可以分为三个区域:栈(stack).堆(heap).方法区(method area): 2.2.1.栈 栈的特点如下: 1.栈描述的 ...
- Android 内存监测工具 DDMS --> Heap(转)
DDMS 的全称是Dalvik Debug Monitor Service,它为我们提供例如:为测试设备截屏,针对特定的进程查看正在运行的线程以及堆信息.Logcat.广播状态信息.模拟电话呼叫.接收 ...
- Java数组之三种初始化及内存分析
内存分析 Java内存 堆: 1.存放new的对象和数组 2.可以被所有的线程共享,不会存放别的对象引用 栈: 1.存放基本变量类型(会包含这个基本类型的具体数值) 2.引用对象的变量(会存放这个引用 ...
- Java实例化对象过程中的内存分配
Java实例化对象过程中的内存分配: https://blog.csdn.net/qq_36934826/article/details/82685791 问题引入这里先定义一个很不标准的“书”类,这 ...
- Day008 三种初始化及内存分析
三种初始化和内存分析 Java内存分析: 堆: 存放new的对象和数组. 可以被所有的线程共享,不会存放别的对象引用. 栈: 存放基本变量类型(会包含这个基本类型的具体数值). 引用对象的变量(会存放 ...
随机推荐
- 将txt转化为csv的方法和遇到问题
一.无法修改扩展名步骤如下 二.转换之后所有数据都挤在第一列 win10系统修改文件扩展名只需4部,打开我的电脑->查看->选择->查看->取消勾选(已知隐藏文件的扩展名)-& ...
- 【WSDL】02 四种客户端调用方式
WSDL概念和一些语法内容: https://www.w3school.com.cn/wsdl/index.asp SOAP概念: https://www.runoob.com/soap/soap-t ...
- 计算机领域:学术写作中的conducive的含义表示
"Conducive" 的意思是"有助于"或"有益于".在学术和正式的写作中,"conducive" 常用于描述某种情况 ...
- 大模型时代该用什么样的显卡 —— 实验室新进两块A800显卡
具体如图: (这两个显卡是专为实验室的大模型方向提供的) 关于A800显卡的性能参数: (上图源自:https://www.zhihu.com/question/618932114/answer/32 ...
- 树莓派3b+ 安装windows10 arm版本的方法及使用体验
首先,我再网上找到了一个很详细的为树莓派3b安装windows10 arm的教程,实际操作下来发现并不可行. 最后找到了可行的教程: 第3章 将Windows10镜像写入TF卡:https://zhu ...
- 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 ...
- vue&element项目实战 之element使用&用户&字典模块实现
6.用户模块 用户模块api import request from '@/utils/request' export function login(data) { return request({ ...
- Sentry For Vue 完整接入详解(2021 Sentry v21.8.x)前方高能预警!三万字,慎入!
内容源于:https://docs.sentry.io/platforms/javascript/guides/vue/ 系列 1 分钟快速使用 Docker 上手最新版 Sentry-CLI - 创 ...
- 结构开发笔记(三):solidworks软件(二):小试牛刀,绘制一个立方体
前言 solidworks草图大师,基本的使用过程. 所有的零件基础都是从平面绘制开始,然后凸出来厚度. 本篇绘制一个简单的立方体,熟悉基本操作. 立方体绘制过程 选取一个平面绘制一个 ...
- bat 随笔
bat 获取文件名 %%~nxi bat 变量去除空字符 BAT批处理中的字符串处理详解(字符串截取)