深入理解JVM - 实战JVM工具(下)

简介: ​ 上一节通过一个APP的JVM内存分析解释了一些比较特殊的参数如何影响JVM,以及分析了之前老年代优化的文章中关于jstat如何进行分析和优化。

前言


接着上篇继续讲述,上一篇模拟了两个还算比较熟悉的场景,分析了之前老年代优化是如何处理的,以及使用jstat分析工具如何分析出JVM的问题,这一节会继续扩展,将会列举更多的案例来分析线上的JVM问题。


前文回顾


上一节通过一个APP的JVM内存分析解释了一些比较特殊的参数如何影响JVM,以及分析了之前老年代优化的文章中关于jstat如何进行分析和优化。


概述:


  1. 介绍三个JVM调优的案例,一步一步分析问题和解决办法。
  2. 总结分析思路和解决流程,自我思考和反思。
  3. 总结和个人感想。


案例实战


案例1:方法区不断崩溃如何排查?


业务场景:


之前的案例基本都是和堆扯上关系,这个案例比较特殊是由 JVM参数设置错误引起的频繁的卡顿问题,简单来说就是设置了参数之后就出现访问系统卡顿并且线上不断的进行FULL GC的报警,这里先不说明改了哪一个参数,而是先来分析一下:


  • 发现系统在十分钟内进行了3次FULL GC,这个频繁十分高
  • 通过线上的JSTAT排查发现出现了报错Methoddata GC Threashold 等字样


根据第二点我们初步断定是JVM的方法区溢出了,**为什么JVM的方法区溢出会触发FULL GC?**事实上确实如此,因为通常情况下FULL GC也会带动方法区的回收。这一块的资料网上可以搜到一大堆,这里不再具体的介绍。


问题分析:


加入验证参数分析:


既然是方法区的问题,为了进一步的排查这个方法区溢出的问题,这里需要加入下面的两个参数:


  • -XX:TraceClassLoading
  • -XX:TraceClassUnloading

这两个参数的作用是追踪类加载类卸载的情况,在加上这两个参数之后继续分析,之后发现在日志的文件当中发现了下面的内容:


image.png


明显可以看到JVM不断的加载了一个叫做GeneratedSerializationCOnstructorAccessor的类,就是这个类不断的加载导致了metaspace区域占满,这也导致了metaspace的对象太多触发FULL GC,所以罪魁祸首就是这个奇怪的类GeneratedSerializationCOnstructorAccessor


为什么会出现奇怪的类?


这里使用google搜索看看这个类是什么,查询结果发现这是JDK内置类,通过查阅资料我们可以知道这是由反射生成的类。


反射的知识点也不再补充,可以理解为一种通过JVM的类加载器结合JVM的工具包生成建立对象的一种方式,也是许多框架的灵魂ss。


其实通过资料查阅我们还可以发现 反射需要JVM动态的生产一些上面所说的奇怪的类到MetaSpace区域,比如要生成动态类ProxyClass@Proxy123123等等类似这种对象(反射生产的类标识比较特殊),JDK都需要上面的辅助对象进行操作。


这里我们还需要在了解一个概念,就是**反射生成的类都是使用的软引用!**至于这个软引用在这里产生了什么影响,这里也先卖个关子,到下文结合把系统搞崩溃的参数一起进行分析。


**什么是软引用?**在内存空间不足的时候被强制回收,不管是否存在局部变量引用。


接着分析关于软引用的存活时间,jvm使用了下面的公式来计算这个软引用的生命周期:


image.png


这个公式的含义:表示一个软引用有多久没有被访问过了,freespace代表了当前的JVM中的空闲内存,softref 代表每一个MB的空间内存空间可以允许SoftReference 存活多久。


估算值:假设现在空间有3000M的对象,softrefLRUpolicyMSPerMB的值为1000毫秒,意味着这些对象会存活 3000秒 也就是50分钟左右。


以上就是奇怪的类出现的原因,是因为反射惹的祸。


排查结果:


到底设置了什么参数?


这里讲一下到底设置了参数,让反射不断生成对象把方法区占满了,这里设置的参数如下:


-XX:SoftRefLRUPolicyMSPerMB=0这个参数。结果JVM就翻车了。


为什么出现奇怪的对象越来越多?


我们再看一下上面的公式:


image.png


假设我们不小心把这个值设置为0有什么后果呢?


当然是会导致上面clock公式的计算结果为0的,结果就是JVM发现每次反射分配的对象马上就会被回收掉,然后接着又会通过代理生成代理对象,简单来说这个参数导致每次soft软引用的对象一旦分配就会马上被回收.


再次强调一下反射机制导致动态代理类不断的被新增,但是这部分对象又被马上回收掉,导致方法区的垃圾对象越来越多这就是为什么奇怪的对象越来越多的原因。


为什么会想着设置这个参数?


设置这个参数的原因也很天真:为了让反射生成的代理对象可以尽快被垃圾回收,如果设置为为0,当方法区的内存占用可以小一些,并且也可以及时回收,然后结果就是好心办坏事


解决办法:


这里解决办法很简单,就是设置一个大于0的值并且最好是1000、2000、5000这一类数字,就是不能设置的过小或者设置为0,否则会导致方法区不断的占用结果方法去溢出最终又导致FULL GC


总结:


这个案例可能看上面的说明很简单就解决了,然而实际上真正碰到类似问题,肯定会出现各种摸不着头脑的情况,希望这篇案例可以让读者设置JVM参数的时候都要验证一下这个参数的影响以及一定要确认他的参数和实际效果是一致的!


这个问题也完全是人的问题,加入没有好奇去想当然的设置一个奇怪的参数,也不至于造成各种奇怪的问题。


最后,只有多学习实际案例,平时多看看别人是如何排查问题的,这对自己的提升也有很大帮助。


案例2:每天数十次GC的线上系统怎么处理?


业务场景:


这个案例和上一个一样是一个实际的案例,话不多说,直接得出当时没优化过的系统的JVM性能表现大致如下:


  • 机器配置:2核4G
  • JVM堆内存大小:2G
  • 系统运行时间:6天
  • 系统运行6天内发生的Full GC次数和耗时:250次,70多秒
  • 系统运行6天内发生的Young GC次数和耗时:2.6万次,1400秒


在使用的时候会发现问题如下:


  • 每天会发生40多次Full GC,平均每小时2次,每次Full GC在300毫秒左右
  • 每天会发生4000多次YGC,每分钟3次,每次YGC在50秒左右。


介绍到这里也可以发现这个系统的性能相当之差,每2小时就会Full GC。这是必须要进行优化的。


优化前JVM参数:


下面是系统优化之前设置的参数:


-Xms1536M -Xmx1536M -Xmn512M -Xss256K -XX:SurvivorRatio=5 -XX:+UseParNewGC -XX:+UseConcMarkSweepGC -XX:CMSInitiatingOccupancyFraction=68 -XX:+CMSParallelRemarkEnabled -XX:+UseCMSInitiatingOccupancyOnly -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintHeapAtGC

我们先不观察其他内容,单从参数本身看下可能会让GC频繁的点:


  1. 可以看到小机器给JVM的堆内存没有多少空间,线程和方法去要分出去一点,JVM的资源十分吃紧。
  2. 新生代明显太小了,而且ratio设置为5,导致最后EDEN区只有可怜的300M左右的空间
  3. 68%的CMS老年代回收阈值似乎有点小,完全可以改为92%才执行回收,老年代的空间比较大
  4. -XX:+CMSParallelRemarkEnabled-XX:+UseCMSInitiatingOccupancyOnly这两个参数的作用请自行百度。


问题分析


1.在后续的排查发现每隔十几分钟就会出现大量的 大对象直接进入老年代,大对象的产生原因是由于开发人员使用的“全表查询”导致了几十万的数据被查出来,这里可以使用jmap的工具进行排查发现生成一个很大的ArrayList,并且内部都是同一个对象。


2.虽然新生代回收之后对象很少的对象进入老年代,几十M,但是可以发现动态规则的判断之后,survior还是有几十M的对象进入到了老年代的空间。


3.新生代的空间很容易饱满,老年代预留空间较大。


4.CMS的阈值设置为68,则达到老年代的68就开始回收,有点过于保守


解决办法:


  1. 如果有条件还是需要加机器,因为机器的性能确实受限。(2G我开IDEA都够呛)
  2. 新生代明显太小了,所以扩大到1G的空间很有必要,同时还是按照5:1:1的分配方案,给survior区域足够的空间和大小。
  3. CMS的回收阈值设置到92%。不需要太过保守。
  4. 方法区指定一个256M的大小,如果不设置任何参数默认的方法区大小只有64M左右的空间


优化之后的参数:


下面是系统优化之后的参数:


-Xms1536M -Xmx1536M -Xmn1024M -Xss256K -XX:SurvivorRatio=5 -XX:PermSize=256M -XX:MaxPermSize=256M  -XX:+UseParNewGC -XX:+UseConcMarkSweepGC -XX:CMSInitiatingOccupancyFraction=92 -XX:+CMSParallelRemarkEnabled -XX:+UseCMSInitiatingOccupancyOnly -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintHeapAtGC

这里再次强调一下上文提到说需要百度的参数:


-XX:+UseCMSInitiatingOccupancyOnly-XX:+CMSParallelRemarkEnabled 这两个参数,这些参数有什么用?


第一个参数:表示每次都在CMS达到设置目标的情况下进行垃圾回收,不会因为JVM动态的判断导致提前进行FULL GC


(这个参数有必要说明一下,因为之前的文章提到过JDK6之后CMS的默认92% 的配比,其实这个配比在实际运行的时候 会根据CMS老年代的回收情况提前或者延迟回收,只能说JVM细节实在是有点多,只要记住开启这个参数之后,CMS会固定真的到达到92%这个比例才进行Full GC垃圾回收的动作


第二个参数:表示进行 并发标记 的步骤之前,先进行一次YGC,其实理论上来说通常都应该进行一次,但是实际上如果不配置JVM会根据实际情况决定是否进行YGC,原因也比较复杂,有兴趣可以把参数复制之后百度补一下课。


案例3:严重的FULL GC导致卡死?


最后介绍一个简单的案例,真的十分简单,3分钟就可以看完:


业务场景:


  1. 一秒一次FULL GC,每次都需要几百毫秒
  2. 平时流量访问不大的时候新生代对象增长不快,老年代占用不到10%,方法区也就20%使用。但是一旦到了高峰时期就是频繁的FULL GC


分析:


在频繁FULL GC的时间点进行GC日志分析,同时使用JMAP分析发现在高峰时期出现大批次操作的对象,这个对象基于一个报表的批量处理的操作,会产生大量的对象并且马上出发回收,结果发现JVM内存放不下导致频繁的FULL GC。


这就很奇怪了我们都知道即使是在平时情况下即使很大数据的批量处理多数情况下并没有离谱到一秒一次FULL GC,那么出现这个问题毫无疑问就是代码的问题了。


经过排查发现,居然有开发人员手动调用垃圾回收也就是System.gc()。这是一个臭名昭著的方法,具体的解释可以看看**《Effective Java》中的第八条**:避免使用终结方法和清除方法.


解决办法:


为了防止System.gc() 生效,这里使用下面的参数禁止掉:


-XX:+DisableExplictGC


总结:


不要写System.gc(),最好是完全不要知道有这个方法。去探究原因其实也是比较浪费时间的事情。


写在最后


JVM的工具实战的上下两篇文章到这里就结束了,后续的文章依然会是实战的部分,从这几个案例可以看到更多的情况下并不是JVM的问题,而是人的问题。


所以 写出优质好理解的代码是本分,写出性能好的代码是水平的体现。先写出好代码才能避免线上出现各种莫名其妙的问题难以排查,而掌握线上的排查和思考手段,可以让个人的能力得到有效的锻炼,所以多多实验和尝试是这篇文章的意义


参考资料:


GeneratedSerializationConstructorAccessor 的资料


资料1:How the sun.reflect.GeneratedSerializationConstructorAccessor class generated 具备特殊上网姿势的建议阅读

答案来自 stackoverflow.com/questions/1…

下面摘自答案的机翻:

第一个回答:


这是因为(可能是您在应用程序中使用反射)堆空间不足,GC 试图通过卸载未使用的对象来释放一些内存,这就是为什么您会看到 Unloading class


sun.reflect.GeneratedSerializationConstructorAccessor


第二个回答


方法访问器和构造器访问器要么是本机的,要么是生成的。这意味着我们对方法使用 NativeMethodAccessorImplGeneratedMethodAccessor,对构造函数使用 NativeConstructorAccessorImplGeneratedConstructorAccessor。访问器可以是原生的或生成的,并由两个系统属性控制和决定:


sun.reflect.noInflation = false(默认值为 false)
sun.reflect.inflationThreshold = 15(默认值为 15)

sun.reflect.noInflation 设置为 true 时,将始终生成所使用的访问器,系统属性 sun.reflect.inflationThreshold 没有任何意义。当 sun.reflect.noInflationfalse 并且 sun.reflect.inflationThreshold 设置为 15 时(如果未指定,这是默认行为)那么这意味着对于构造函数(或方法)的前 15 次访问,本机生成器将被使用,此后将提供一个生成的访问器(来自 ReflectionFactory)以供使用。


Native 访问器使用本地调用来访问信息,而生成的访问器都是字节码,因此速度非常快。另一方面,生成的访问器需要时间来实例化和加载(基本上是膨胀,因此控制它的系统属性的名称包括“膨胀”一词)。


更多细节可以在原始博客中找到.....

相关文章
|
1月前
|
NoSQL Java Redis
秒杀抢购场景下实战JVM级别锁与分布式锁
在电商系统中,秒杀抢购活动是一种常见的营销手段。它通过设定极低的价格和有限的商品数量,吸引大量用户在特定时间点抢购,从而迅速增加销量、提升品牌曝光度和用户活跃度。然而,这种活动也对系统的性能和稳定性提出了极高的要求。特别是在秒杀开始的瞬间,系统需要处理海量的并发请求,同时确保数据的准确性和一致性。 为了解决这些问题,系统开发者们引入了锁机制。锁机制是一种用于控制对共享资源的并发访问的技术,它能够确保在同一时间只有一个进程或线程能够操作某个资源,从而避免数据不一致或冲突。在秒杀抢购场景下,锁机制显得尤为重要,它能够保证商品库存的扣减操作是原子性的,避免出现超卖或数据不一致的情况。
63 10
|
2月前
|
Arthas Prometheus 监控
监控堆外使用JVM工具
监控堆外使用JVM工具
54 7
|
2月前
|
存储 IDE Java
实战优化公司线上系统JVM:从基础到高级
【11月更文挑战第28天】Java虚拟机(JVM)是Java语言的核心组件,它使得Java程序能够实现“一次编写,到处运行”的跨平台特性。在现代应用程序中,JVM的性能和稳定性直接影响到系统的整体表现。本文将深入探讨JVM的基础知识、基本特点、定义、发展历史、主要概念、调试工具、内存管理、垃圾回收、性能调优等方面,并提供一个实际的问题demo,使用IntelliJ IDEA工具进行调试演示。
53 0
|
3月前
|
JavaScript 前端开发 Java
jvm的jshell,学生的工具
本文介绍了JVM的jshell工具,它为Java平台添加了REPL(读取-评估-打印循环)功能,使得学习、探索编码和原型代码变得更加便捷,但作者认为其在实际开发中较为鸡肋。
54 1
jvm的jshell,学生的工具
|
3月前
|
Arthas 监控 数据可视化
JVM进阶调优系列(7)JVM调优监控必备命令、工具集合|实用干货
本文介绍了JVM调优监控命令及其应用,包括JDK自带工具如jps、jinfo、jstat、jstack、jmap、jhat等,以及第三方工具如Arthas、GCeasy、MAT、GCViewer等。通过这些工具,可以有效监控和优化JVM性能,解决内存泄漏、线程死锁等问题,提高系统稳定性。文章还提供了详细的命令示例和应用场景,帮助读者更好地理解和使用这些工具。
|
3月前
|
监控 架构师 Java
JVM进阶调优系列(6)一文详解JVM参数与大厂实战调优模板推荐
本文详述了JVM参数的分类及使用方法,包括标准参数、非标准参数和不稳定参数的定义及其应用场景。特别介绍了JVM调优中的关键参数,如堆内存、垃圾回收器和GC日志等配置,并提供了大厂生产环境中常用的调优模板,帮助开发者优化Java应用程序的性能。
|
3月前
|
存储 监控 算法
JVM调优深度剖析:内存模型、垃圾收集、工具与实战
【10月更文挑战第9天】在Java开发领域,Java虚拟机(JVM)的性能调优是构建高性能、高并发系统不可或缺的一部分。作为一名资深架构师,深入理解JVM的内存模型、垃圾收集机制、调优工具及其实现原理,对于提升系统的整体性能和稳定性至关重要。本文将深入探讨这些内容,并提供针对单机几十万并发系统的JVM调优策略和Java代码示例。
74 2
|
3月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
68 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
5月前
|
缓存 监控 算法
吃透 JVM 诊断方法与工具使用
【8月更文挑战第4天】深入了解并掌握JVM诊断需把握几大要点:1) 熟悉JVM内存模型,如堆、栈及方法区;2) 掌握垃圾回收机制与算法;3) 运用工具如`jps`(查看Java进程)、`jstat`(监控运行状态)、`jmap`(生成堆快照)、`jhat`(分析堆快照)、`jstack`(检查线程栈); 4) 利用专业工具如Eclipse Memory Analyzer分析堆转储文件查找内存泄漏; 5) 动态监控与调整JVM参数; 6) 结合日志分析性能瓶颈。通过实战案例加深理解,有效应对JVM性能问题。
|
5月前
|
存储 监控 算法
深入解析JVM内部结构及GC机制的实战应用
深入解析JVM内部结构及GC机制的实战应用