Android OOM问题定位、内存优化
一、OOM
out of memory:简称OOM,内存溢出,申请的内存大于剩余的内存而抛出的异常。
对于Android平台,广义的OOM主要是以下几种类型
- Java
- Native
- Thread
线程数的上限默认为32768,部分华为设备的限制是500通常1000左右就会触发VSS OOM。因此Thread 00M其实也是VSS OOM的一种表现形式。
引起线程OOM原因:1.文件描述符不够 2.地址空间不足
- File Descriptor
Android 9.0以前fd也是比较宝贵的资源,每个进程的上限只有1024,9.0开始增加到了32768。
- JNI Reference
JNl Reference 0OM可以通过logcat初步定位。
常用工具:
1、LeakCanary
val refWatcher: RefWatcher? = TestApp.getRefWatcher(activity)
refWatcher?.watch(activity);//检测是否有泄露,即触发GC回收,看activity是否被回收,没有被回收就是泄露了。
二、常见的几种内存泄漏
1、长生命周期引用短生命周期2、资源没有释放
单例造成的内存泄露
非静态内部类创建静态实例造成的内存泄露
Handler造成的内存泄露
线程造成的内存泄露
Webview造成的内存泄露
三、OOM的原因,以及如何定位OOM
OOM产生的原因:
1、一次性申请的太多
更改申请对象数量
2.内存资源耗尽未释放
找到未释放的对象进行释放
3.本身资源不够
jmap -heap 查看堆信息
如何优化内存
1、减小对象的内存占用
2、内存对象的复用
3、避免对象的内存泄露
如何定位OOM
如何通过dump定位
系统已经OOM挂了
提前设置-XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=
系统运行中还未OOM
导出dump文件:jmap -dump:format=b,file=xushu.hprof 14660Arthas
结合ivisualvm 进行调试
查看最多跟业务有关对象->找到GCRoot ->查看线程栈
四、GC回收原理
GCRoot——对象可达性分析
可回收对象的判定
通过一系列称为“GC Roots”的对象作为起始点,从这些节点向下搜索,搜索所有的引用链,当一个对象到GC Roots没有任何引用链(即GC Roots到对象不可达)时,则证明此对象是不可用的。

五、Java内存分配模型

GCRoot是:方法区Method Area、虚拟机栈VM Stack(即本地变量)、本地方法栈Native Method(即静态变量和常量) Stack中的变量。如GCRoot中变量有对堆中对象的引用,gc则不回收堆中的该对象,如堆中对象无来自GCRoot的引用,则回收该对象。例如,B b = new B();对象b为引用存于虚拟机栈VM Stack中,new出的B对象存于堆中。
六、Android profiler——dump内存快照分析
双击打开设备Downloads文件夹下hprof文件,文件信息在profiler中被显示。
shallow size浅堆和Retain size深堆的区别
shallow size(浅堆)是指对象本身占用内存的情况
Retain size(深堆)是指对象以及对象关联其他引用占用内存的总和。即,对象本身被GC回收后,导致其他对象同时也被GC回收,这些被回收的所有对象占用内存总和。
如下举例:
1、对象A,B,C内存占用情况如下,A的shallow size是10M,Retain size是30M

1、对象A,B,C,D内存占用情况如下
A的shallow size是10M,Retain size是20M
B的shallow size是10M,Retain size是10M
C的shallow size是10M,Retain size是10M
D的shallow size是10M,Retain size是10M

七、Java的四种引用
1.强引用(Strong Reference)在代码中普遍使用的,类似Person person=new Person();如果一个对象具有强引用,则无论在什么情况下,GC都不会回收被引用的对象。当内存空间不足时,JAVA虚拟机宁可抛出outOfMemoryError终止应用程序也不会回收具有强引用的对象。
2.软引用(Soft Reference)表示一个对象处在有用但非必须的状态。如果一个对象具有软引用,在内存空间充足时,GC就不会回收该对象;当内存空间不足时,GC会回收该对象的内存(回收发生在0utofMemoryError之前)。软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被GC回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中,以便在恰当的时候将该软引用回收。但是由于GC线程的优先级较低,通常手动调用system.gc()并不能立即执行GC,因此弱引用所引用的对象并不一定会被马上回收。
3.弱引用(Weak Reference)用来描述非必须的对象。它类似软引用,但是强度比软引用更弱一些:弱引用具有更短的生命,GC在扫描的过程中,一旦发现只具有被弱引用关联的对象,都会回收掉被弱引用关联的对象。换言之,无论当前内存是否紧缺,GC都将回收被弱引用关联的对象。
4.虚引用(Phantom Reference)虚引等同于没有引用,这意味着在任何时候都可能被GC回收,设置虚引用的目的是为了被虚引用关联的对象在被垃圾回收器回收时,能够收到一个系统通知。(被用来跟踪对象被GC回收的活动)虚引用和弱引用的区别在于:虚引用在使用时必须和引用队列(ReferenceQueue)联合使用,其在GC回收期间的活动如下:ReferenceQueue queue=new ReferenceQueue();PhantomReference pr=new PhantomReference(obiect.queue)也即是GC在回收一个对象时,如果发现该对象具有虚引用,那么在回收之前会首先该对象的虚引用加入到与之关联的引用队列中。程序可以通过判断引用队列中是否已经加入虚引用来了解被引用的对象是否被GC回收。
软引用和弱引用的区别如下:
软引用:gc扫到,不一定回收,内存不足时才回收
弱引用:gc扫到,就回收。
弱引用和引用队列的关系如下:
泄露了:弱引用没有加入引用队列,故引用队列为空
没有泄露:弱引用加入引用队列,故引用队列不为空
八、Java新生代,老年代和永久代的区别
在Java虚拟机(JVM)的内存管理中,堆内存通常被划分为几个不同的区域,以便更有效地管理和回收内存。以下是新生代(Young Generation)、老年代(Old Generation)和永久代/元空间(PermGen/Metaspace)的具体含义:
1. 新生代 (Young Generation)
- 新生代是Java堆内存的一部分,主要用于存储新创建的对象。对象在首次分配内存时会被放入新生代。
- 新生代的特点是大量对象在此快速生成并很快消亡,因此这里的垃圾回收(Garbage Collection, GC)非常频繁,通常称为Minor GC或Young GC。
- 新生代进一步细分为 Eden 区、两个 Survivor 区(例如:From 和 To 区),每次GC后,存活下来的对象会被复制到另一个Survivor区,若对象经历了一定次数的GC还存活,则晋升至老年代。
2. 老年代 (Old Generation)
- 老年代也是Java堆内存的一部分,主要存放经过一定时间周期仍然存活下来的对象,即经历过多次新生代GC仍然没有被回收的对象。
- 对象一旦晋升到老年代,意味着它们生命周期较长,老年代的垃圾回收相对较少,但是一旦发生,通常会是Major GC或Full GC,这类GC会比Minor GC更加耗时且影响更大。
3. 永久代 (PermGen)
- 在JDK 8及之前版本的HotSpot JVM中,永久代是方法区的一个实现,用于存储类信息、常量池、静态变量、方法字节码以及其他运行时常驻数据结构。
- 永久代的空间有限,如果加载了大量的类或者反射操作过于频繁,可能导致永久代空间不足,从而抛出`java.lang.OutOfMemoryError: PermGen space`异常。
4. 元空间 (Metaspace)
:
- 自JDK 8开始,HotSpot JVM取消了永久代的概念,并引入了一个新的内存区域叫做元空间(Metaspace)。
- 元空间同样用于存储类元数据信息,但它位于本地内存(Native Memory)而非堆内存中,这意味着它的大小不再受到JVM堆大小的限制,而受限于系统的实际可用内存。
- 当类元数据的总量超过了指定阈值或者系统内存限制时,将会触发类卸载机制,并可能出现`java.lang.OutOfMemoryError: Metaspace`异常。
总结起来,新生代和老年代是用来区分对象生命周期长短进而进行高效内存回收的堆内存区域,而永久代在旧版JVM中是方法区的一种实现方式,存储类和方法级别的元数据;在新版JDK中,这部分功能由元空间替代。
九、JVM出现OOM异常会导致进程挂掉吗?
当一个线程在执行代码的过程中,大概率需要创建对象,而创建对象就需要分配内存,如果VM可用内存不够时会进行垃圾回收,如果垃圾回收完了之后内存还是不够就会地出OutOfMemoryEror,如果没有捕获OutOfMemoryError,那么就像抛出一个普通异常一样会导致线程停掉,如果捕获了OutOfMemoryEror,那么线程可能就不会亭掉,其实不管当前线程会不会停掉,跟进程会不会挂掉没有直接关系,也就是出现,了OutOfMemgrvEror最多只会导致线程停掉,如果一个讲程里面的所有非守护线程都亭掉了,那么进程才会停掉,或者进程占了操作系统的过多内存,那么这个进程可能会被操作系统关闭掉。
参考:https://blog.csdn.net/pengweid/article/details/137125832
参考:彻底解决Android开发中的OOM问题,竟然一节课就够了_哔哩哔哩_bilibili
参考:美团面试题:一个线程 OOM 后,其他线程还能运行吗?_哔哩哔哩_bilibili
相关文章:
Android OOM问题定位、内存优化
一、OOM out of memory:简称OOM,内存溢出,申请的内存大于剩余的内存而抛出的异常。 对于Android平台,广义的OOM主要是以下几种类型 JavaNativeThread 线程数的上限默认为32768,部分华为设备的限制是500通常1000左右…...
棋盘(c++题解)
题目描述 有一个m m的棋盘,棋盘上每一个格子可能是红色、黄色或没有任何颜色的。你现在要从棋盘的最左上角走到棋盘的最右下角。 任何一个时刻,你所站在的位置必须是有颜色的(不能是无色的) ,你只能向上、下、 左、右…...
滑动窗口例题
一、209:长度最小的子数组 209:长度最小的子数组 思路:1、暴力解法:两层for循环遍历,当sum > target时计算子数组长度并与result比较,取最小的更新result。提交但是超出了时间限制。 class Solution {public int minSubArray…...
智过网:注册安全工程师注册有效期与周期解析
在职业领域,各种专业资格认证不仅是对从业者专业能力的认可,也是保障行业安全、规范发展的重要手段。其中,注册安全工程师证书在安全生产领域具有举足轻重的地位。那么,注册安全工程师的注册有效期是多久呢?又是几年一…...
腐蚀Rust 服务端搭建架设个人社区服务器Windows教程
腐蚀Rust 服务端搭建架设个人社区服务器Windows教程 大家好我是艾西,一个做服务器租用的网络架构师也是游戏热爱者。最近在steam发现rust腐蚀自建的服务器以及玩家还是非常多的,那么作为服务器供应商对这商机肯定是不会放过的哈哈哈! 艾西这…...
蓝桥杯备赛:考前注意事项
考前注意事项 1、DevCpp添加c11支持 点击 工具 - 编译选项 中添加: -stdc112、万能头文件 #include <bits/stdc.h>万能头文件的缺陷:y1 变量 在<cmath>中用过了y1变量。 #include <bits/stdc.h> using namespace std;// 错误示例 …...
111111111111
111111111111...
uniapp 卡片勾选
前言 公司的app项目使用的uniapp,项目里有一个可勾选的卡片功能,效果图如下: 找了一圈没找到什么太好的组件,于是就自己简单写了一个,记录一下。避免以后还会用到 代码 <template><view class"card-…...
乐趣Python——文件与数据:挥别乱糟糟的桌面
各位朋友们,今天我们要开启一场非凡的冒险——进入文件操作的世界!你知道吗,在你的电脑里,有一个叫做“文件系统”的迷宫,里面藏着各种各样的文件和文件夹,它们就像是迷宫中的宝藏。但有时候,这…...
docker nginx-lua发送post json 请求
环境准备 dockerfile from fabiocicerchia/nginx-lua:1.25.3-ubuntu22.04 run apt-get -qq update && apt-get -qq install luarocks run luarocks install lua-cjson run luarocks install lua-iconv run luarocks install lua-resty-http后台代理服务准备ÿ…...
阿里面试总结 一
写了这些还是不够完整,阿里 字节 卷进去加班!奥利给 ThreadLocal 线程变量存放在当前线程变量中,线程上下文中,set将变量添加到threadLocals变量中 Thread类中定义了两个ThreadLocalMap类型变量threadLocals、inheritableThrea…...
多线程(49)定义无锁、阻塞、非阻塞和无等待算法
在并发编程中,理解不同的同步策略——无锁(Lock-Free)、阻塞(Blocking)、非阻塞(Non-Blocking)、无等待(Wait-Free)——对于设计高效、健壮的多线程应用至关重要。让我们…...
(一)ffmpeg 入门基础知识
一、ffmpeg FFmpeg是一套强大的开源音视频处理工具,能够录制、转换以及流化音视频内容。 FFmpeg是开源的,这意味着它的源代码是公开的,允许任何人使用、修改和分发。它提供了录制、转换以及流化音视频的完整解决方案,支持多种格…...
【软件测试】个人博客系统测试
个人博客系统测试 一、项目背景1.1 技术背景1.2 功能背景 二、 测试用例编写三、自动化测试3.1 什么是自动化测试3.2 通过使用selenium进行自动化测试的编写(Java实现)3.3 编写测试用例,执行自动化测试3.3.1 输入用户名:test,密码:123&#x…...
20240410解决OK3588-C的核心板刷机之后无法启动的问题
20240410解决OK3588-C的核心板刷机之后无法启动的问题 2024/4/10 19:38 1、编译OK3588的LINUX/Buildroot?forlinxubuntu: ~/3588/OK3588_Linux_fs$ sudo ./build.sh BoardConfig-linuxfs-ok3588.mk 2、进行全编译 forlinxubuntu: ~/3588/OK3588_Linux_fs$ sudo ./bu…...
仅需三步就能成为大语言模型Prompt Engineer提示词工程大神
AI Prompt Engineer(提示词工程)是当下GenAI行业最热门的话题,它是利用有效的AI模型交互提示技术,引导大语言模型生成更高质量、更准确、更相关的回应。相对于预训练和微调,提示词工程不需要标注数据和训练模型,极大的节约了时间和…...
RuleEngine规则引擎底层改造AviatorScript 之公式规则
前情提要,看上一个文章,具体要实现的效果就是 当然上来的问题就是前端的问题,这个框首先他们用的是富文本,富文本传到后台的结果是前端脚本,带着h5的标签,后面改成了这个,当时这个东西其实和后…...
Vue项目(H5)与微信小程序来回跳转
新建H5页面 在小程序里面新建一个名为H5的文件夹,以及H5页面 H5.WXML <web-view src"{{h5Url}}" bindmessage"handleGetMessage"></web-view>H5.JSdata: { h5Url:https://xxx.com/login 要跳转的H5页面},H5回来的回调方法handleG…...
设计模式-单一职责原则
基本介绍 对类来说的,即一个类应该只负责一项职责。如类A负责两个不同的职责,职责1,职责2.当职责1需求变更而改变A时,可能造成职责2执行错误,所以需要将类A的粒度分解为A1,A2 应用实例 方案1 public cl…...
vue和nunjucks的变量插值的形式{{}}冲突
Nunjucks 中修改配置 const nunjucks require(nunjucks);const template_old nunjucks.renderString(template_old: Hello, {{name}}!, { name: World }); console.log(template_old); // 配置 Nunjucks 环境 nunjucks.configure({tags: {variableStart: $(, // 设置变量起始…...
Windows下FFmpeg环境配置全攻略:从下载到视频剪辑实战
Windows下FFmpeg环境配置全攻略:从下载到视频剪辑实战 在数字内容创作爆发的时代,视频处理能力已成为开发者和创作者的必备技能。FFmpeg作为开源多媒体处理领域的"瑞士军刀",其强大功能与跨平台特性使其成为处理音视频文件的首选工…...
基于Matlab的转子系统临界转速与主振型求解:传递矩阵法及其参数涉及等截面、材料与轮盘参数的...
140.基于matlab的求解转子系统前三个临界转速和主振型的传递矩阵法转子系统的不平衡响应 参数涉及等截面参数、材料参数、轮盘参数 程序已调通,可直接运行传递矩阵法这玩意儿在转子动力学里属于实操性极强的工具,今天咱们就拿Matlab直接开搞转子系统的前…...
SEO_详解SEO核心关键词的研究与布局方法(455 )
<h2>SEO核心关键词的研究与布局方法详解</h2> <p>在当前的互联网时代,搜索引擎优化(SEO)已经成为了各个企业和网站提升网络曝光率、吸引更多流量的重要手段。其中,核心关键词的研究与布局是SEO的重要组成部分。…...
DALL-E2-pytorch训练日志完全解读指南:如何从loss曲线判断模型健康状态
DALL-E2-pytorch训练日志完全解读指南:如何从loss曲线判断模型健康状态 【免费下载链接】DALLE2-pytorch Implementation of DALL-E 2, OpenAIs updated text-to-image synthesis neural network, in Pytorch 项目地址: https://gitcode.com/gh_mirrors/da/DALLE2…...
前端性能优化终极指南:使用Javalin实现静态资源压缩与智能缓存
前端性能优化终极指南:使用Javalin实现静态资源压缩与智能缓存 【免费下载链接】javalin 项目地址: https://gitcode.com/gh_mirrors/jav/javalin 在现代Web应用开发中,前端资源的加载速度直接影响用户体验和搜索引擎排名。Javalin作为一款轻量级…...
GB28181实战:Windows环境下WVP-GB28181部署全攻略
1. Windows环境下WVP-GB28181部署全攻略 如果你正在寻找一个在Windows系统上快速搭建GB28181视频监控平台的方法,那么WVP-GB28181绝对是个不错的选择。作为一个开源的视频监控平台,WVP-GB28181支持国标GB/T28181协议,能够帮助你轻松实现视频设…...
IOMMU性能调优全攻略:从基础原理到实战技巧
IOMMU性能调优全攻略:从基础原理到实战技巧 在数据中心和云计算环境中,IOMMU(输入输出内存管理单元)作为硬件辅助虚拟化的关键技术组件,其性能表现直接影响着整个系统的吞吐量和延迟。对于需要处理高并发I/O负载的场景…...
零基础图解VLN视觉语言导航:从输入到决策的完整模型拆解
1. 视觉语言导航(VLN)是什么? 想象你第一次去朋友家做客,对方在电话里说:“进门左转,看到红色沙发后直走,右手边第二个房间就是。”这时候你的大脑会做三件事:用眼睛观察环境&#x…...
SEO_从入门到精通,掌握SEO的核心操作步骤
<h2>SEO从入门到精通,掌握SEO的核心操作步骤</h2> <p>在当今的互联网时代,搜索引擎优化(SEO)已经成为任何网站或网页希望获得高流量、高曝光的关键技能。无论你是一个初学者,还是已经有一些SEO基础的…...
5分钟玩转OpenClaw:nanobot镜像云端体验与本地调试对比
5分钟玩转OpenClaw:nanobot镜像云端体验与本地调试对比 1. 为什么需要对比云端与本地两种体验方式 作为一个长期折腾AI工具的开发者,我最近在测试OpenClaw时遇到了一个典型困境:是直接在本地电脑安装全套环境,还是先用云端沙盒快…...
