记一次项目内存优化--内存泄漏

需求–内存泄漏优化,PSS有所下降, OOM率减少

主要是与某个版本作基准进行对比(一般是最新版本的前一个版本作原数据),优化后,PSS有所下降,线上OOM率减少(Bugly版本对比),泄漏点减少(从捉取一些线上上传回来的内存堆栈信息分析,或本地测试后dump下hprof文件分析)。

内存泄漏优化的思路

  • 了解什么是内存泄漏
  • 了解虚拟机中的对象的创建过程
  • 了解Java内存分配模型
  • 了解垃圾回收分代收集理论
  • 了解java的引用类型
  • GC是如何判断对象存活
  • 有哪些对象可作为GC Roots
  • 了解内存泄漏的工具
  • 总结

什么是内存泄漏

App程序中己动态分配的堆内存,由于某种原因,App程序未释放或无法释放,会造成系统(手机)内存的浪费。长生命周期对象持有短生命周期对象强引用,从而导致短生命周期对象无法被回收。 我们注意这两个关键词堆内存、强引用

虚拟机中的对象的创建过程(类的生命周期)

什么都不用说,先上张自画图。为大家推荐一本书《深入理解JVM》

第一步,当虚拟机遇到一条new指令时,首先检查这条指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有,那必须先执行相应的类加载过程。

第二步,如果检查通过后,虚拟机将为这个new出的对象进行分配内存。划分内存是通过指针碰撞、空间列表的组合,同时也考虑并发安全问题(CAS(Compare And Swap的缩写–乐观锁)失败重试、本地线程内存缓冲)。这中是进行内存分配哦,这时候还不能确定对象所需要的内存大小。在类加载完成后才确定内存的大小。

第三步,内存分配完成后,虚拟机将分配到的内存空间进行初始化为零值(默认的初始值),但不包括对象头信息,如果使用TLAB(Thread Local Allocation Buffer ,即线程本地分配缓冲区),这过程可以提前至TLAB分配时进行(Eden区划分出一小块区域作为TLAB)。这一步操作是保证了对象的实例成员(字段)在Java代码中可以不赋初始值就直接使用,程序能访问到这些字段的数据类型所对应的零值。

第四步,对象进行必要的设置(主要是一些对象头信息的设置),例如这个对象的运行状态、GC分代年龄、锁状态、属于哪个类的实例、哈希码等等信息。

第五步,从前面几个步骤知道,只是设置了对象头信息和所有类成员(字段)赋为默认的初始值,对象并没有执行方法,所以最后是会接着执行方法,这样才算创建出一个真正可用的对象。几乎所有对象是存放在堆区。

Java内存分配模型

使用一张图,快速了解一下内存分配模型。

方法区

编译时就分配好,在程序整个运行期间都存在。它用于存储已经被虚拟机加载的类信息、静态变量、常量等数据。

堆区

几乎所有 new 出来的对象是存放在堆区,由 Java 垃圾回收器回收。堆中的对象是垃圾回收的重点。 堆区的划分新生代、老年代:

新生代

新生代是用来存放新new出来的对象,划分为 Eden区、From Survivor区 、To Survivor区 。几乎所以的new出来的对象都会存放在Eden区 (如果new出来的对象占用的内存非常大,在新生代中存放不下,直接进入老年代存放)。

当Eden区的内存空间不足时,系统会触发Minor GC /Young GC进行回收Eden区的对象(From Survivor区 、To Survivor区不会触发GC),经过GC后,一些对象仍然存活(对象被引用着–通过GC Root可达性来判断),则会被移到To Survivor区存放,当对象在Survivor区熬过一次GC后,此对象的GC年龄就会+1(GC年龄是对象头信息的一个标记参数),会被复制到From Survivor区,当From Survivor区的对象达到一定年龄时(默认年龄是15,但可以通过XX:MaxTenuringThreshold设置),被移到老年代,否则复制到To Survivor区。

老年代

老年代是新生代存放不下的大对象,或对象经过多次Minor GC /Young GC后仍然存活的对象(长期存活的对象)。

当随着Eden区的Minor GC /Young GC持续进行,老年代的对象持续增加,会导致老年代可用的内存空间也会持续减少,最终系统会触发Major GC。

元空间(永久代)

永久代(持久代)是存放包含应用的类/方法信息,以及JRE库的类和方法信息。然而在Java8中,元空间取代了永久代,元空间(Metaspace)被称为“元数据区”。

需要注意的是:元空间并不在虚拟机中哦,而是使用本地内存(以前永久代是在jvm中的)。这样就解决了以前永久代的OOM问题,元数据和class对象存放在永久代中,容易出现性能问题和内存溢出,毕竟是和老年代共享堆空间。

堆内存分配策略

内存分配原则

  • 对象优先在Eden分配----如果说Eden内存空间不足,就会发生Minor GC /Young GC。
  • 大对象直接进入老年代----大对象:需要大量连续内存空间的Java对象,比如很长的字符串和大型数组。会导致新生代内存有空间,还是需要提前进行垃圾回收获取连续空间来放此大对象。Survivor区会进行大量的内存复制,-XX:PretenureSizeThreshold 参数 ,大于这个数量直接在老年代分配,缺省为0 ,表示绝不会直接分配在老年代。当Eden分配和Survivor区都没有足够空间存放此大对象时,则直接分配到老年代。
  • 长期存活的对象将进入老年代----Survivor区的对象达到一定年龄时,直接移到老年代。默认15岁,可以通过XX:MaxTenuringThreshold设置。
  • 动态对象年龄判定----为了能更好地适应不同程序的内存状况,虚拟机并不是永远地要求对象的年龄必须达到了MaxTenuringThreshold才能晋升老年代,如果在Survivor区中相同年龄所有对象大小的总和大于Survivor区的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,无须等到MaxTenuringThreshold中要求的年龄。
  • 空间分配担保:新生代中有大量的对象存活,Survivor区不够,当出现大量对象在Minor GC后仍然存活的情况(最极端的情况就是内存回收后新生代中所有对象都存活),就需要老年代进行分配担保,把Survivor区无法容纳的对象直接进入老年代,只要老年代的连续空间大于新生代对象的总大小或者历次晋升的平均大小,就进行Minor GC,否则Full GC。

栈中分配对象

  • 逃逸分析----如果符合逃逸分析规则,则在栈中分配对象。

堆中的优化技术

  • TLAB ----Thread Local Allocation Buffer ,即线程本地线程分配缓冲。

栈区

当方法执行时,会在栈区内存中创建方法体内部的局部变量,方法结束后自动释放内存。

垃圾回收分代收集理论

我们都知道,在java中不同的对象存在不同的生命周期的,java对象在JVM中也存放在不同的区域,所以对不同生命周期不同的存放区,采取不同的回收策略,以提高效率。

当Eden区的内存空间不足时,系统触发Minor GC/Young GC, 随着GC持续进行,老年代的对象持续增加,导致老年代的内存空间不足,系统触发Major GC。当堆区或方法区内存空间不足时,系统触发Full GC。

Full GC:清理成本高,系统资源消耗高,对系统性能产生影响,很多性能什么都是针对Full GC进行的。 触发Full GC的条件有:

  • 调用System.gc()
  • 方法区空间不足
  • 堆区空间不足

不同阶段GC的特点

  • Minor GC/Young GC – 执行非常频繁,速度特别快。
  • Major GC – 速度上,一般会比Minor GC/Young GC慢十倍以上。
  • Full GC – Minor GC和Major GC都会执行,会发出"Stop the World"事件,会中断程序运行,直到GC完成。所以Full GC时,我们会感知到APP有卡顿之感。

垃圾回收分代收集对应的回收算法

  • 复制算法: 实现简单,运行高效,内存复制,内存利用率只有一半。
  • 标记-清除: 利用率100%,不需要内复制,有内存碎片
  • 标记-整理:利用率100%,没有内存碎片,需要内存复制(整理存活的对象,将其拷贝到一块连续内存中)

GC是如何判断对象存活

  • 可达性分析 (java) 通过一系列称之为“GC Roots”的对象作为起始点,从这些节点向下搜索,搜索所有的引用链,当一个对象到GC Roots有引用链,则说明这个对象存活着;当一个对象到GC Roots没有任何引用链(即GC Roots到对象不可达)时,则证明此对象是不可用的(所谓的垃圾)。

  • 引用计数算法(JVM早期使用的—已经不使用) A对象引用B 对象(+1),同时C对象引用B对象(1+1=2),计数法就是引用一次累加1次,如果没有引用就累减1次,如果归到0时,说明没有引用。缺点:就是相互引用。如A对象引用B对象,同时B对象引用A对象,很难去判断对象是否应该回收。

在Java, 可作为GC Roots的对象包括:

  • 方法区: 类静态属性的对象;
  • 方法区: 常量的对象;
  • 虚拟机栈(本地变量表)中的对象。
  • 本地方法栈JNI(Native方法)中的对象。

四种引用类型

  1. 强引用(StrongReference):JVM 宁可抛出 OOM ,也不会让 GC 回收具有强引用的对象。
  2. 软引用(SoftReference):只有在内存空间不足时,对象才会被回收。
  3. 弱引用(WeakReference):在 GC 时,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,对象都会被回收。
  4. 虚引用(PhantomReference):任何时候都可以被GC回收,当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。程序可以通过判断引用队列中是否存在该对象的虚引用,来了解这个对象是否将要被回收。可以用来作为GC回收Object的标志。

我们定义对象,应该考虑使用那种引用,多考虑使用软引用(定义一些还有用但并非必须的对象)或弱引用(定义非必须对象)。

Android Studio的profiler工具

  1. 我们也可以利用Android Studio的profiler工具,方便快速查找、观察,简单分析一些对象生成情况。
  2. 也可以dump下hprof文件,结合MAT深入分析与排查,对象发生是否泄漏。
  3. 注意:MAT打开Android Studio的profiler里dump下hprof文件时,利用AS自带的hprof工具转换一下格式(通过命令hprof-conv -z 原hprof文件 输出hprof文件),不然打开是乱码。

当然检测内存泄漏的工具和方法有很多,就不一一列举了,感兴趣的可以网上查阅一下。

常见的内存问题场景

  • 静态成员/单例
    • 作为GC ROOT,持有短生命周期引用(如Activity)导致其短生命周期对象无法释放。
  • 集合类
    • 当使用集合时,只有添加元素,没有对应的删除元素。
  • 非静态内部类/匿名内部类
    • 如Handler postDelayed一个匿名Runnable,退出Activity时消息没处理完。
  • 上下文 – Context
    • 持有的上下文,需要特别注意。
  • 注册/反注册
    • 如EventBus只有注册没有注销。addXXXListener函数,需要有对应的removeXXXListener等等。
  • 未关闭/释放资源
    • 如FileOutputStream未close。
  • 系统Bug
    • WebView、InputMethodManager等

总结

  • 上面的内存相关知识也是自己学习的一种总结,有错误的可以留言指正。
  • 内存优化,需要对下面的知识有一定的了解。
    • Java内存分配模型
    • Java的四大引用及其使用场景
    • 内存检测工具及常用命令
    • GC Root的定义

为了帮助到大家更好的全面清晰的掌握好性能优化,准备了相关的核心笔记(还该底层逻辑):https://qr18.cn/FVlo89

性能优化核心笔记:https://qr18.cn/FVlo89

启动优化

内存优化

UI优化

网络优化

Bitmap优化与图片压缩优化https://qr18.cn/FVlo89

多线程并发优化与数据传输效率优化

体积包优化

《Android 性能监控框架》:https://qr18.cn/FVlo89

《Android Framework学习手册》:https://qr18.cn/AQpN4J

  1. 开机Init 进程
  2. 开机启动 Zygote 进程
  3. 开机启动 SystemServer 进程
  4. Binder 驱动
  5. AMS 的启动过程
  6. PMS 的启动过程
  7. Launcher 的启动过程
  8. Android 四大组件
  9. Android 系统服务 - Input 事件的分发过程
  10. Android 底层渲染 - 屏幕刷新机制源码分析
  11. Android 源码分析实战

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/85226.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Unsafe upfileupload

文章目录 client checkMIME Typegetimagesize 文件上传功能在web应用系统很常见,比如很多网站注册的时候需要上传头像、上传附件等等。当用户点击上传按钮后,后台会对上传的文件进行判断 比如是否是指定的类型、后缀名、大小等等,然后将其按…

Php“牵手”淘宝商品SKU信息数据采集方法,淘宝API接口申请指南

淘宝天猫商品属性sku信息接口 API 是开放平台提供的一种 API 接口,它可以帮助开发者获取商品的详细信息,包括商品的标题、描述、图片,销量,sku信息等信息。在电商平台的开发中,商品属性接口API是非常常用的 API&#x…

JS中对象数组深拷贝方法

structuredClone() JavaScript 中提供了一个原生 API 来执行对象的深拷贝:structuredClone。它可以通过结构化克隆算法创建一个给定值的深拷贝,并且还可以传输原始值的可转移对象。 当对象中存在循环引用时,仍然可以通过 structuredClone()…

基本定时器

1.简介 1. 基本定时器 TIM6 和 TIM7 包含一个 16 位自动重载计数器 2. 可以专门用于驱动数模转换器 (DAC), 用于触发 DAC 的同步电路 3. 16 位自动重载递增计数器 4. 16 位可编程预分频器 5. 计数器溢出时, 会触发中断/DMA请求 从上往下看 1.开始RCC供给定时器的时钟 RCC_APB1…

通过Matlab编程分析微分方程、SS模型、TF模型、ZPK模型的关系

微分方程、SS模型、TF模型、ZPK模型的关系 一、Matlab编程 微分方程、SS模型、TF模型、ZPK模型的关系二、对系统输出进行微分计算三、对系统输出进行积分计算四、总结五、系统的零点与极点的物理意义参考 :[https://www.zhihu.com/question/22031360/answer/3073452…

【AGC】Publishing api怎么上传绿色认证审核材料

【问题描述】 华为应用市场会对绿色应用标上特有的绿色标识,代表其通过华为终端开放实验室DevEco云测平台的兼容性、稳定性、安全、功耗和性能的检测和认证,是应用高品质的象征。想要自己的应用认证为绿色应用就需要在发布应用时提供绿色认证审核材料&a…

数据结构之——(手撕)顺序表

本章会介绍的知识点如下图: 1: 顺序表的概念:顺序表是用一段物理地址连续的存储单元依次存储数据的线性结构,通常我们使用数组来表示,对数组进行增删查改。 顺序表的结构:逻辑结构与物理结构都是内存中一块…

【宝藏系列】一文讲透C语言数组与指针的关系

【宝藏系列】嵌入式 C 语言代码优化技巧【超详细版】 文章目录 【宝藏系列】嵌入式 C 语言代码优化技巧【超详细版】👨‍🏫前言1️⃣指针1️⃣1️⃣指针的操作1️⃣2️⃣关于指针定义的争议1️⃣3️⃣对教材错误写法的小看法 2️⃣指针和数组的区别2️⃣…

lab5 lazy

文章目录 Eliminate allocation from sbrk()Lazy allocationtaskhints实现 Lazytests and Userteststaskhints实现 Eliminate allocation from sbrk() 第一个任务是去阻止sysproc.c中的sys_sbrk()函数真的分配内存,只需要增p->sz即可 一行代码注释即可 uint64…

pandas数据分析40——读取 excel 合并单元格的表头

案例背景 真的很容易疯....上班的单位的表格都是不同的人做的,所以就会出现各种合并单元格的情况,要知道我们用pandas读取数据最怕合并单元格了,因为没规律...可能前几列没合并,后面几列又合并了....而且pandas对于索引很严格&am…

VR数字工厂多元化展现,打造数字企业工厂名片

5G时代,各种营销都在走数字化的路子,VR数字工厂用VR赋能工厂数字升级,将企业环境、工厂生产、产品研发、质检运输等流程,无死角720度的展示在客户面前,不仅可以提升自身企业的实力,还可以提高客户的信任感。…

使用Pandas处理Excel文件

Excel工作表是非常本能和用户友好的,这使得它们非常适合操作大型数据集,即使是技术人员也不例外。如果您正在寻找学习使用Python在Excel文件中操作和自动化内容的地方,请不要再找了。你来对地方了。 在本文中,您将学习如何使用Pan…

超级计算机

超级计算机是一种高性能计算机,它能够以极高的速度执行大规模的计算任务。超级计算机通常由数千个甚至数百万个处理器组成,这些处理器能够同时处理大量的数据,从而实现高效的计算。超级计算机广泛应用于科学、工程、金融、天气预报等领域&…

5G与4G的RRC协议之异同

什么是无线资源控制(RRC)? 我们知道,在移动通信中,无线资源管理是非常重要的一个环节,首先介绍一下什么是无线资源控制(RRC)。 手机和网络通过无线信道相互通信,彼此交…

SpringBoot - 两种方式刷新配置信息

一、第一种方式 ​ConfigurationProperties​不能自动刷新,需要手动调用contextRefresher.refresh()方法来刷新配置。 import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.stereotype.Component;Component…

C#学习....

1.基础 //引用命名空间using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;//项目名或者命名空间 namespace _01_MY_First_Demo {//Program类class Program{//程序的主入口或者Main函数static void Main(S…

前端开发怎么解决性能优化的问题? - 易智编译EaseEditing

前端性能优化是确保网站或应用在加载速度、响应性和用户体验等方面达到最佳状态的关键任务。以下是一些解决前端性能优化问题的方法: 压缩和合并代码: 压缩和合并CSS、JavaScript和HTML文件可以减少文件大小,加快加载速度。使用压缩工具&am…

分布式核心知识以及常见微服务框架

分布式中的远程调用 在微服务架构中,通常存在多个服务之间的远程调用的需求。远程调用通常包含两个部分:序列化和通信协议。常见的序列化协议包括json、xml、 hession、 protobuf、thrift、text、 bytes等,目前主流的远程调用技术有基于HTTP…

C语言编写图形界面

文章目录 环境使用库基础概念句柄 程序的入口创建窗口定义窗口类注册窗口类创建窗口 完整代码运行效果 环境 使用的是VSCode MinGW&#xff1b; 使用库 我们使用windows.h库来实现图形化界面。 头文件如下&#xff1a; #include <windows.h>windows.h是 Windows 操作…

特斯拉Model 3的七年狂飙

‍ 作者 | 张祥威 编辑 | 德新 发布一周拿下32万张订单&#xff0c;之后用时五年&#xff0c;交付量突破100万辆。粗略计算&#xff0c;自2016年发布至今&#xff0c;特斯拉Model 3已交付超150万辆。 放眼新能源赛道&#xff0c;如此战绩 别无二家。 Model 3踩中纯电动车的…
最新文章