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: $(, // 设置变量起始…...
网络六边形受到攻击
大家读完觉得有帮助记得关注和点赞!!! 抽象 现代智能交通系统 (ITS) 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 (…...
谷歌浏览器插件
项目中有时候会用到插件 sync-cookie-extension1.0.0:开发环境同步测试 cookie 至 localhost,便于本地请求服务携带 cookie 参考地址:https://juejin.cn/post/7139354571712757767 里面有源码下载下来,加在到扩展即可使用FeHelp…...
Docker 离线安装指南
参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性,不同版本的Docker对内核版本有不同要求。例如,Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本,Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...
【kafka】Golang实现分布式Masscan任务调度系统
要求: 输出两个程序,一个命令行程序(命令行参数用flag)和一个服务端程序。 命令行程序支持通过命令行参数配置下发IP或IP段、端口、扫描带宽,然后将消息推送到kafka里面。 服务端程序: 从kafka消费者接收…...
大话软工笔记—需求分析概述
需求分析,就是要对需求调研收集到的资料信息逐个地进行拆分、研究,从大量的不确定“需求”中确定出哪些需求最终要转换为确定的“功能需求”。 需求分析的作用非常重要,后续设计的依据主要来自于需求分析的成果,包括: 项目的目的…...
Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...
Springboot社区养老保险系统小程序
一、前言 随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱,社区养老保险系统小程序被用户普遍使用,为方…...
华硕a豆14 Air香氛版,美学与科技的馨香融合
在快节奏的现代生活中,我们渴望一个能激发创想、愉悦感官的工作与生活伙伴,它不仅是冰冷的科技工具,更能触动我们内心深处的细腻情感。正是在这样的期许下,华硕a豆14 Air香氛版翩然而至,它以一种前所未有的方式&#x…...
七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...
20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
