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

需求–内存泄漏优化,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.mzph.cn/news/42214.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

程序员如何利用公网远程访问查询本地硬盘【内网穿透】

🎬 鸽芷咕:个人主页 🔥 个人专栏: 《高效编程技巧》《cpolar》 ⛺️生活的理想,就是为了理想的生活! 公网远程访问本地硬盘文件【内网穿透】 文章目录 公网远程访问本地硬盘文件【内网穿透】前言1. 下载cpolar和Everything软件1.…

(stm32)低功耗模式

低功耗模式 执行哪个低功耗模式的程序判断流程 标志位设置操作一定要在WFI/WFE之前,调用此指令后立即进入睡眠判断流程 模式对比 睡眠模式 停止模式 待机模式

FLatten Transformer

FLatten Transformer: Vision Transformer using Focused Linear Attention ICCV 2023 聚焦式线性注意力模块 关于Transformer 在Transformer模型应用于视觉领域的过程中,降低自注意力的计算复杂度是一个重要的研究方向。线性注意力通过两个独立的映射函数来近似S…

3 Python的数据类型

概述 在上一节,我们介绍了Python的基础语法,包括:编码格式、标识符、关键字、注释、多行、空行、缩进、引号、输入输出、import、运算符、条件控制、循环等内容。Python是一种动态类型的编程语言,这意味着当你创建一个变量时&…

react 生命周期方法

组件的生命周期 每个组件都包含 “生命周期方法”,你可以重写这些方法,以便于在运行过程中特定的阶段执行这些方法。你可以使用此生命周期图谱作为速查表。在下述列表中,常用的生命周期方法会被加粗。其余生命周期函数的使用则相对罕见。 挂…

Windows Oracle21C与PLSQL Developer 15配置

1、下载Oracle21c并安装 下载地址:https://www.oracle.com/database/technologies/oracle21c-windows-downloads.html 2、下载PLSQL Developer 15并安装 下载地址:https://www.allroundautomations.com/products/pl-sql-developer/#pricing 3、配置O…

在线课堂录播直播管理系统SpringBoot+Vue

在线课堂录播直播管理系统SpringBootVue 文章目录 在线课堂录播直播管理系统SpringBootVue共三个端:后端、后台管理系统、前端,如要学习看评论区(全部源码、文档、数据库)。内置功能一、前端二、后台管理三、后端--代码全有。四、…

数据结构—排序

8.排序 8.1排序的概念 什么是排序? 排序:将一组杂乱无章的数据按一定规律顺序排列起来。即,将无序序列排成一个有序序列(由小到大或由大到小)的运算。 如果参加排序的数据结点包含多个数据域,那么排序往…

async和await

一,基本使用 其实就是之前学过的异步函数,异步编程在函数前写一个ansyc,就转化为异步函数,返回的是一个promise对象,于是就可以使用await关键字,可以把异步函数写成同步函数的形式,极大地提高代…

LVS-DR的RS进行ARP抑制的原因和LVS持久连接配置

一.RS的ARP抑制 1.为什么要抑制 2.如何抑制 (1)修改/etc/sysctl.conf文件,增加以下内容 (2)命令行临时设置 二.LVS持久连接 1.客户端持久连接 2.端口持久连接 3.防火墙标记持久连接 一.RS的ARP抑制 1.为什么要…

学校信息管理系统说明文档

目录 0学生信息管理系统体验教程. 4 0.0Student management异地打开方法:. 4 1. 管理系统设计需求分析. 6 1.1 需求介绍. 6 1.2功能需求. 6 1.2.1 学生信息录入. 6 1.2.2 学生信息查询. 6 1.2.3 权限管理. 6 1.2.4 添加学生信息验证. 6 2.功能介绍. 7 2.1…

快速上手PyCharm指南

PyCharm简介 PyCharm是一种Python IDE(Integrated Development Environment,集成开发环境),带有一整套可以帮助用户在使用Python语言开发时提高其效率的工具,比如调试、语法高亮、项目管理、代码跳转、智能提示、自动…

idea如何建立web项目???

我们需要用到tomcat,没有下在着小伙伴,可以借鉴这篇博客: 如何正确下载tomcat???_明天更新的博客-CSDN博客 1.建立普通的Java项目。 2.简单编写index.jsp文件 3.添加tomcat 4.运行服务器 5.构建Servlet 最后…

嵌入式编译FFmpeg6.0版本并且组合x264

下载直通车:我用的是6.0版本的 1.准备编译: 2.进入ffmpeg源码目录,修改Makefile,添加编译选项: CFLAGS -fPIC 不加会报错 3.使用命令直接编译 ./configure --cross-prefix/home/xxx/bin/arm-linux-gnueabihf- --enable-cross-compile --targ…

CodeSite for .NET Crack

CodeSite for .NET Crack CodeSite for.NET与Visual Studio集成,通过实时查看器日志记录系统提供对代码执行的更深入了解,该系统有助于在本地或远程执行代码时快速查找问题。超越传统的断点调试,在应用程序继续运行时记录应用程序的执行&…

vue使用jsplumb 流程图

安装jsPlumb库&#xff1a;在Vue项目中使用npm或yarn安装jsPlumb库。 npm install jsplumb 创建一个Vue组件&#xff1a;创建一个Vue组件来容纳jsPlumb的功能和呈现。 <template><div style"margin: 20px"><div style"margin: 20px">&l…

Python学习笔记_基础篇(八)_正则表达式

1. 正则表达式基础 1.1. 简单介绍 正则表达式并不是Python的一部分。正则表达式是用于处理字符串的强大工具&#xff0c;拥有自己独特的语法以及一个独立的处理引擎&#xff0c;效率上可能不如str自带的方法&#xff0c;但功能十分强大。得益于这一点&#xff0c;在提供了正则…

Spring系列篇 -- Bean的生命周期

目录 经典面试题目&#xff1a; 一&#xff0c;Bean的生命周期图 二&#xff0c;关于Bean的生命周期流程介绍&#xff1a; 三&#xff0c;Bean的单例与多例模式 总结&#xff1a; 前言&#xff1a;今天小编给大家带来的是关于Spring系列篇中的Bean的生命周期讲解。在了解B…

DAY06_SpringBoot—简介基础配置yaml多环境开发配置整合第三方技术

目录 一 SpringBoot简介1. 入门案例问题导入1.1 入门案例开发步骤1.2 基于SpringBoot官网创建项目1.3 SpringBoot项目快速启动 2. SpringBoot概述问题导入2.1 起步依赖2.2 辅助功能 二 基础配置1. 配置文件格式问题导入1.1 修改服务器端口1.2 自动提示功能消失解决方案1.3 Spri…

国产化系统中遇到的视频花屏、卡顿以及延迟问题的记录与总结

目录 1、国产化系统概述 1.1、国产化操作系统与国产化CPU 1.2、国产化服务器操作系统 1.3、当前国产化系统的主流配置 2、视频解码花屏与卡顿问题 2.1、视频解码花屏 2.2、视频解码卡顿 2.3、关于I帧和P帧的说明 3、国产显卡处理速度慢导致图像卡顿问题 3.1、视频延…