JVM内存变化分析实战

news2024/11/24 23:01:10

最近在一次项目压力测试时,监测到JVM内存明显的变化,由于之前开发工作中没有涉及到JVM相关的问题分析,所以特此借这个机会学习和记录。项目使用的JDK版本为 OpenJdk 1.8,虚拟机为 HotSpot

1. 内存变化情况

在压力测试进行2H48Min时,JVM内存出现明显变化,内存最大占用量为1.53G,最大波动量在800MB左右,具体图示如下

image.png

2. 内存变化情况

2.1 元空间变化情况

元空间.png

在JDK1.8中,方法区是使用 元空间 来实现的。元空间使用的是 本地内存 ,其中存储有 类型信息常量静态变量即时编译器编译后的代码缓存 等数据。 运行时常量池 也是方法区的一部分,它存放的是Class文件中在编译期生成的常量池表和运行期间新生成的常量。从上图中可以发现,该内存区域使用内存大小没有发生改变,所以方法区与本次JVM内存变化情况无关。

2.2 年轻代内存变化情况

年轻代内存变化.png

年轻代伊甸园区占用内存从544MB减少到94MB,变化了450MB;年轻代Survivor区从0MB增加到129MB并又随后减少到0MB。

2.3 老年代内存变化情况

老年代.png

老年代内存从104MB增加到1.12GB,之后经历老年代GC又减少到831MB,变化量约为727MB,与堆内存总变化量接近。

3. 内存变化分析

GC.png

3.1 年轻代GC分析

PS Scavenge 表示新生代使用的是 Parallel Scavenge 垃圾收集器,它采用的是 标记-复制 算法。标记-复制算法在对年轻代进行收集时,采用复制分代策略,将年轻代划分为伊甸园区和两个Survivor区,比例为8:1:1,本次项目JVM分配的堆内存大小为2G,那么内存大小比例为1638.4M:204.8M:204.8M,算法执行时会直接将Eden和其中一个Survicor区中存活的对象复制到另一个Survivor区,如果Survivor区不足以保存存活的对象,那么会触发 逃生门 机制直接将对象移至老年代。

从年轻代内存变化图示中可以发现,对年轻代进行垃圾回收时,Survivor区从0M变化到129M。0M说明之前Survivor区并没有被使用,如果因此推测每次GC没有对象存活的话,那么显然是不准确的,因为没有对象存活那么怎么才能导致老年代内存变化,难道是因为一次性创建了800M的大对象直接分配到了老年代(不可能)?

如果考虑标记复制算法因存在 逃生门 机制,导致Survivor区长时间是0M,那么应该在能够在Survivor区0M使用的时间里观察到老年代有明显的内存升高才对(对象大小超过204.8M直接分配到老年代),但是在老年代内存变化图示中没有相关变化。

在这里已经陷入了困惑,猜测可能是因为监控数据每15s采集一次而每次GC的平均时间在500ms,导致没有监控到Survicor区的内存变化而产生的这种情况。可能实际上Survicor区有被使用,即每次GC后有对象存活,否则老年代内存的升高从何而来?而且我们能够发现老年代内存变化除较大波动的范围外,其他时间是一条斜率较小的线,也证明着有小对象随着垃圾回收不断的往老年代中加入,不过本次内存的较大波动无从分析。

而且图示中Survivor区增加到129M,这个数据很可能是不够准确的。在图示中可以发现申请的(Committed)内存大小最大有223M之多,与监控到的最大使用内存129M相差近100M,理论上申请内存大小与使用内存应该比较接近才对,所以在这里也能发现监控数据的异常。

3.2 老年代GC分析

从图示中可以发现老年代GC是 PS Mark Sweep ,表示 标记-清除 算法,以此推断老年代使用的是 CMS 收集器。它是一种以获取最短停顿时间为目的收集器,会经历初始标记、并发标记、重新标记和并发清除四个阶段,图示中GC耗时在2s左右。CMS在启动时会默认要求分配 (核心处理数 + 3) / 4 个核数,本次压测服务器核数为32,那么在执行回收时会分配8核左右,GC完成后老年代内存从1.12GB减少到831M。

3.3 总结

基于不准确的数据判断和推测:JVM堆内存变化是年轻代中的对象在进行垃圾回收时被转移到老年代导致的,但是没有准确的数据支撑这个结论看起来非常牵强,而且还有一点需要注意:项目服务端使用Netty搭建,调用本地方法在本地内存中生成的对象虽然不直接占用堆内存,但是它也会有在Java堆里的 DirectByteBuffer 对象作为这块内存的引用,不清楚这会不会造成堆内存的升高。

由于本次压测的关注重点并不在此,而在于内存和CPU的最大使用情况,所以之后并没有调整数据采集时间间隔再进行压测,如果之后再进行压力测试应注意该参数的设置。


巨人的肩膀

  • 《深入理解Java虚拟机(第三版)》:第二章、第三章

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

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

相关文章

Java008——Java关键字和标识符的简单认识

一、Java关键字 围绕以下3点介绍: 1、什么是Java关键字? 2、Java有哪些关键字? 3、Java关键字的作用? 4、Java关键字的使用?后面文章再做介绍 1.1、什么是Java关键字? 定义:被Java语言赋予了…

github开源化课程体系推荐 浙江大学 计算机考研必备408资料汇总 北京大学计算机系资料整理

github漫游指南 github漫游指南 *所有开源课程资料网站整理在文末 什么是GitHub Wiki 百科上是这么说的 GitHub 是一个共享虚拟主机服务,用于存放使用Git版本控制的软件代码和内容项目。它由GitHub公司(曾称Logical Awesome)的开发者Chr…

【手撕Spring源码】深度理解SpringMVC【下】

文章目录 控制器方法执行流程ControllerAdvice 之 ModelAttribute返回值处理器MessageConverterControllerAdvice 之 ResponseBodyAdviceBeanNameUrlHandlerMapping 与 SimpleControllerHandlerAdapterRouterFunctionMapping 与 HandlerFunctionAdapterSimpleUrlHandlerMapping…

Elasticsearch:节点角色 - node roles

你可能已经知道 Elasticsearch 集群由一个或多个节点组成。 每个节点将数据存储在分片上,每个分片存储在一个节点上。 到目前为止,你看到的每个节点都至少存储了一个分片,但值得注意的是,节点并不总是必须存储分片。 这是因为每个…

【Unity3D】运动模糊特效

1 运动模糊原理 开启混合(Blend)后,通过 Alpha 通道控制当前屏幕纹理与历史屏幕纹理进行混合,当有物体运动时,就会将当前位置的物体影像与历史位置的物体影像进行混合,从而实现运动模糊效果,即模…

javascript基础二十二:举例说明你对尾递归的理解,有哪些应用场景

一、递归 递归(英语:Recursion) 在数学与计算机科学中,是指在函数的定义中使用函数自身的方法 在函数内部,可以调用其他函数。如果一个函数在内部调用自身本身,这个函数就是递归函数 其核心思想是把一个大型…

Redis高级数据结构之GEO

GEO的介绍 Redis3.2版本提供了GEO地址位置信息定位的功能。支持存储地理位置信息来实现诸如摇一摇,附近位置这类地理位置信息的功能。 Redis也是使用业界比较通用的地理位置距离排序算法GeoHash算法。将二维的经纬度坐标数据映射到一维的整数,将所有元素…

1.项目环境部署操作

第一步 将资料中提供虚拟机压缩包, 解压到一个没有中文没有空格, 以及磁盘空间相对充足的磁盘中(大于100GB) 第二步 修改VMware的网卡设置: 统一修改为 88网段, 网关为192.168.88.2 vm虚拟机 windows系统 第三步 将两台项目虚拟机挂载到VMware上

Kubernetes学习笔记-开发应用的最佳实践(2)20230604

三、确保所有的客户端请求都得到了妥善处理 如何在pod启动的时候,确保所有的连接都被妥善处理了 1、在pod启动时避免客户端连接断开 当个pod启动的时候,他以服务端点的方式提供给所有的服务,这些服务的标签选择器和pod的标签匹配。pod需要…

【简单实用框架】【读Excel表】【可移植】

☀️博客主页:CSDN博客主页 💨本文由 我是小狼君 原创,首发于 CSDN💢 🔥学习专栏推荐:面试汇总 ❗️游戏框架专栏推荐:游戏实用框架专栏 ⛅️点赞 👍 收藏 ⭐留言 📝&…

【算法思维】-- 贪心算法

OJ须知: 一般而言,OJ在1s内能接受的算法时间复杂度:10e8 ~ 10e9之间(中值5*10e8)。在竞赛中,一般认为计算机1秒能执行 5*10e8 次计算。 时间复杂度取值范围o(log2n)大的离谱O(n)10e8O(nlog(n))10e6O(nsqrt(…

LinuxC编程——标准IO

目录 标准I/O一、概念二、特点⭐⭐⭐三、缓冲区⭐⭐⭐3.1 全缓冲3.1 行缓冲3.3 不缓冲 四、函数接口⭐⭐⭐⭐4.1 打开4.1.1 fopen4.1.2 freopen4.1.2 容错机制perror 4.2 关闭4.2.1 fclose4.3 读写操作4.3.1 字符I/O4.3.2 行I/O4.3.3 块I/O 4.4 定位操作4.5 文件结束和错误 脑图…

Redis高级数据结构之HyperLogLog

HyperLogLog的介绍 这并不是一种全新的数据结构、实际类型是一种字符串类型。通过一种基数(不重复的元素数量就是基数)算法,便可以使用很小的内存空间完成独立总数的统计。数据集可以是IP、Email、ID等官方给出的统计误差是0.81%&#xff0c…

python文本注释数学表达式设置|python绘图中的数学表达式设置

本篇文章将介绍如何在Matplotlib中设置文本、注释和数学表达式,以便更好地呈现数据,提高可视化效果。 文章目录 一、Matplotlib中的文本设置1.1 纯文本设置1.2 含箭头的文本设置 二、Matplotlib中的数学表达式设置三、Matplotlib中的字体设置 一、Matplo…

Vue项目中通过插件pxtorem实现大屏响应式

一、原理 rem单位代表的是根节点的font-size大小,所以当我们在页面上使用rem去替代px的时候,就可以通过修改根节点font-size的值,动态地让页面上的元素根据不同浏览器宽高下去实现变化。 二、工具 1.postcss-pxtorem 作用:在编…

Spring Boot Application.properties和yaml配置文件

文章目录 一、全局配置文件概述二、Application.properties配置文件(一)创建Spring Boot的Web项目PropertiesDemo(二)在应用属性文件里添加相关配置1、配置服务器端口号和web虚拟路径2、对象类型的配置与使用(1&#x…

微应用如何实现自动更新提示

首先, 先讲一下本次文章所讲的场景, 经过调研, 公司内部使用后台, 当有需求功能迭代的时候, 通常使用者会没有感知, 使用者只会在浏览器内一直打开这个页面, 当需要使用的时候, 再切换这个tab来使用. 这就导致使用者一直不知道系统更新了, 一直没有访问最新的页面(由于最新页面…

日志框架——Log4j2

日志框架——Log4j2 日志框架Log4j21. 概述2. Log4j2主要由几个重要的组件构成:3.项目中使用3.1 引入相关依赖pom.xml3.2 加入日志配置文件src/main/resources/log4j2.xml3.3 测试 日志框架Log4j2 1. 概述 Apache Log4j2是一个开源的日志记录组件,使用非常的广泛。…

【Protobuf速成指南】什么是Protobuf?

文章目录 一、序列化和反序列化1.1 概念1.2 场景1.3 如何序列化 二、Protobuf介绍1. 自身特点2.使用特点 一、序列化和反序列化 1.1 概念 🎯[总结]: 序列化:把对象转换为字节序列的过程称为对象的序列化。反序列化:把字节序列恢复为对象的过…

MySQL数据库 8.DML操作

目录 ​编辑 🤔前言: 🤔DML介绍: 🤔语法详情: 😀1.插入数据: 特点: 1.给指定字段添加数据: 代码示例: 运行结果: 2.给所有的字段添加数据&…