每年大促都要补习JVM调优,今年我都总结到这里了。

举报
西魏陶渊明 发表于 2022/09/25 00:49:53 2022/09/25
【摘要】 作者: 西魏陶渊明 博客: https://blog.springlearn.cn/ 天下代码一大抄, 抄来抄去有提高, 看你会抄不会抄! 示例代码地址 一、JVM内存介绍 我们...

作者: 西魏陶渊明
博客: https://blog.springlearn.cn/

天下代码一大抄, 抄来抄去有提高, 看你会抄不会抄!

示例代码地址

一、JVM内存介绍

我们在学习JVM的内存管理的时候,我们的思维要跳出Java的局限。我们要这么理解。我们写的Java代码,是运行在JVM上的。
如果让你来实现JVM那么。你会怎么处理呢?

  • 公共部分(堆heap)
    • Class字节码是公共的,是共享的,所有线程都要认识字节码。
    • new的对象是公共的,也是共享的,所有线程要都能认识这些实例对象,能读取到实例的数据。
  • 私有部分 (栈stock)
    • Java中每个线程的执行中的代码,及代码中的局部变量等信息是私有的。每个线程之间都要维护一份。
    • JVM虚拟栈和本地方法栈。
    • 代码是怎么执行的,当然是一行一行执行。那么这一行一行的代码是放在哪里的呢? 是放在栈里面的。Java代码是在JVM来执行的。
      所以这个栈,我们称为JVM虚拟栈
    • JVM中有些方法是调用其他语言实现的, 会使用本地方法栈
    • 那么谁来读取栈里面的数据,来出栈执行呢? 这叫做PC寄存区

1.1 堆空间

JVM参数配置说明

1.1.1 堆上信息

new 出来的对象都在堆上。当堆的内存不足,会触发gc。GC策略

1.1.2 堆的相关配置

配置参数 说明 示例
-Xmx 设置最大堆大小。 -Xmx3550m,设置JVM最大可用内存为3550 MB。
-Xms 设置JVM初始内存。 -Xms3550m,设置JVM初始内存为3550 MB。此值建议与-Xmx相同,避免每次垃圾回收完成后JVM重新分配内存。
-Xmn2g 设置年轻代大小。 -Xmn2g,设置年轻代大小为2 GB。整个JVM内存大小=年轻代大小+年老代大小+持久代大小。持久代一般固定大小为64 MB,所以增大年轻代后,将会减小年老代大小。此值对系统性能影响较大,Sun官方推荐配置为整个堆的3/8。
-XX:NewRatio=n 设置年轻代和年老代的比值。 -XX:NewRatio=4,设置年轻代(包括Eden和两个Survivor区)与年老代的比值(除去持久代)。如果设置为4,那么年轻代与年老代所占比值为1:4,年轻代占整个堆栈的1/5。
-XX:SurvivorRatio=n 年轻代中Eden区与两个Survivor区的比值。 -XX:SurvivorRatio=4,设置年轻代中Eden区与Survivor区的大小比值。如果设置为4,那么两个Survivor区与一个Eden区的比值为2:4,一个Survivor区占整个年轻代的1/6。
-XX:MaxPermSize=n 设置持久代大小。(JDK8以移除) -XX:MaxPermSize=16m,设置持久代大小为16 MB。
-XX:MaxTenuringThreshold=n 设置垃圾最大年龄。 -XX:MaxTenuringThreshold=0,设置垃圾最大年龄。如果设置为0,那么年轻代对象不经过Survivor区,直接进入年老代。对于年老代比较多的应用,提高了效率。如果将此值设置为较大值,那么年轻代对象会在Survivor区进行多次复制,增加了对象在年轻代的存活时间,增加在年轻代即被回收的概率。

1.1.3 常见问题

  • 大对象,无法释放,导致内存移除。
    堆上的问题是比较容易排查的,可以通过工具把堆的信息给dump下来,然后就能直接定位到大对象,并通过调用链路定位到具体的代码,后面会介绍工具

1.2 非堆空间

1.2.1 非堆上的信息

  • Thread 配置线程的栈大小,决定了你调用链的深度。
  • Metaspace 可加载类的信息大小

1.2.2 相关配置

配置参数 说明 示例
-Xss 设置线程的栈大小。 -Xss128k,设置每个线程的栈大小为128 KB。说明 JDK 5.0版本以后每个线程栈大小为1 MB,JDK 5.0以前版本每个线程栈大小为256 KB。请依据应用的线程所需内存大小进行调整。在相同物理内存下,减小该值可以生成更多的线程。但是操作系统对一个进程内的线程个数有一定的限制,无法无限生成,一般在3000个~5000个。
-XX:MaxMetaspace=n 设置元空间大小。 -XX:MaxMetaspace=16m,设置元空间大小为16 MB。

二、工具介绍

2.1 原生命令

2.2 二方可视化分析

2.2.1 idea 插件VisualGC

2.2.2 JProfile

JProfile

2.2.3 Arthas

Arthas

Arthas功能是比较强大的,非常适合用于排查些疑难问题

2.3 GC日志学习

  • 开启GC日志参数 -XX:+PrintGCDateStamps -XX:+PrintGCDetails -XX:+PrintGCDateStamps

2.3.1 年轻代GC

  • 首先是年轻代GC
2022-06-03T00:13:48.801-0800: 
0.369: 
[GC (Allocation Failure) 
[PSYoungGen: 7168K->1513K(8704K)] 7168K->4097K(49664K), 0.0183816 secs] 
[Times: user=0.02 sys=0.01, real=0.02 secs] 

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 2022-06-03T00:13:48.801-0800 -XX:+PrintGCDateStamps 打印日期
  • 0.369 -XX:+PrintGCDateStamps JVM启动到当前日期的总时长的时间戳形式
  • [GC (Allocation Failure) GC 原因(Allocation Failure) 分配失败
    • Allocation Failure 分配失败
    • Metadata GC Threshold 元空间不足
    • Last ditch collection 元空间GC后,仍然不足,即触发
  • PSYoungGen 年轻代GC
  • Times 耗时统计
    • user 表示GC线程执行所使用的CPU总时间
    • sys 进程在内核态消耗的CPU时间
    • real 程序从开始到结束所用的时钟时间,这个时间接近 sys + user

由于多核的原因,一般的GC事件中, real time是小于sys + user time的,因为一般是多个线程并发的去做GC,所以real time是要小于systuser time的

2.3.2 老年代GC

老年代执行的是 Full GC,Full GC执行的时候,不止回收老年代,还会回收新生代和元数据空间

2022-06-03T00:22:27.829-0800:
0.798: 
[Full GC (Allocation Failure) 
[PSYoungGen: 0K->0K(8704K)] 
[ParOldGen: 36024K->36006K(40960K)] 36024K->36006K(49664K), 
[Metaspace: 3078K->3078K(1056768K)], 0.2006976 secs] 
[Times: user=1.11 sys=0.01, real=0.21 secs] 

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 2022-06-03T00:13:48.801-0800 -XX:+PrintGCDateStamps 打印日期
  • 0.369 -XX:+PrintGCDateStamps JVM启动到当前日期的总时长的时间戳形式
  • [Full GC (Allocation Failure) GC 原因(Allocation Failure) 分配失败
    • Allocation Failure 分配失败
    • Metadata GC Threshold 元空间不足
    • Last ditch collection 元空间GC后,仍然不足,即触发
  • PSYoungGen 年轻代GC
  • ParOldGen 老年代GC
  • Metaspace 元空间或者叫方法区GC
  • Times 耗时统计
    • user 表示GC线程执行所使用的CPU总时间
    • sys 进程在内核态消耗的CPU时间
    • real 程序从开始到结束所用的时钟时间,这个时间接近 sys + user

三、场景分析

3.1 堆空间导致OOM

3.1.1 模拟堆栈

  • HeapOverflowTest
  • StackOverflowTest

3.1.2 现象

  1. 频繁进行fu gc
  2. 应用吞吐量下降
  3. 应用rt上升
  4. 方法调用报错OutOfMemoryError : Java heap space

3.1.3 解决方案

  1. jps 找到应用 pid
  2. 把堆信息dump下来 jmap -dump:format=b,file=heap.hprof ${pid}
  3. 打开JProfile 打开文件,直接看到大对象是哪个。

3.2 CPU飙升

CPU飙升,可能是有线程一直在占用CPU。发生了死锁,发生了死循环之类的。这些情况是有问题的。
但是当你的机器流量比较大时候,同样也会导致CPU飙升,此时可能就需要加机器来进行解决。或者仅限限流。下面
只说有问题的场景,如何查看线程状态。

3.2.1 模拟异常

public class CPU {

    public static void main(String[] args) {
        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                while (true) {

                }
            }).start();
        }
    }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

3.2.2 现象

  • 系统卡顿,吞吐量下降
  • 如果没有限制启动参数,可能会导致宿主机也非常卡段,引用占用了很大CPU

3.2.3 解决方案

  1. 找到那些线程在阻塞 jstack $PID
  2. 如下片段发现线程都是 BLOCKED 状态, 调用点都在 CPU.java:18
"Thread-497" #508 prio=5 os_prio=31 tid=0x00007f88f58a0000 nid=0x41903 waiting for monitor entry [0x0000000326ea5000]
   java.lang.Thread.State: BLOCKED (on object monitor)
	at java.io.PrintStream.println(PrintStream.java:735)
	- waiting to lock <0x00000007bce02720> (a java.io.PrintStream)
	at learn.jvm.CPU.lambda$main$0(CPU.java:18)
	at learn.jvm.CPU$$Lambda$1/189568618.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)

"Thread-496" #507 prio=5 os_prio=31 tid=0x00007f88f589f800 nid=0x41a03 waiting for monitor entry [0x0000000326da2000]
   java.lang.Thread.State: BLOCKED (on object monitor)
	at java.io.PrintStream.println(PrintStream.java:735)
	- waiting to lock <0x00000007bce02720> (a java.io.PrintStream)
	at learn.jvm.CPU.lambda$main$0(CPU.java:18)
	at learn.jvm.CPU$$Lambda$1/189568618.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)

"Thread-495" #506 prio=5 os_prio=31 tid=0x00007f8905034000 nid=0x41c03 waiting for monitor entry [0x0000000326c9f000]
   java.lang.Thread.State: BLOCKED (on object monitor)
	at java.io.PrintStream.println(PrintStream.java:735)
	- waiting to lock <0x00000007bce02720> (a java.io.PrintStream)
	at learn.jvm.CPU.lambda$main$0(CPU.java:18)
	at learn.jvm.CPU$$Lambda$1/189568618.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23

::: tip Arthas
当然如果你安装了 Arthas

你可以 thread -n 3 打印出最忙的三个线程 thread.
直接输入cpu使用量

thread -b, 找出当前阻塞其他线程的线程
:::

"Thread-8" Id=19 cpuUsage=89.17% deltaTime=188ms time=17319ms RUNNABLE
    at learn.jvm.CPU.lambda$main$0(CPU.java:13)
    at learn.jvm.CPU$$Lambda$1/500977346.run(Unknown Source)
    at java.lang.Thread.run(Thread.java:748)


"Thread-1" Id=12 cpuUsage=85.62% deltaTime=180ms time=17296ms RUNNABLE
    at learn.jvm.CPU.lambda$main$0(CPU.java:13)
    at learn.jvm.CPU$$Lambda$1/500977346.run(Unknown Source)
    at java.lang.Thread.run(Thread.java:748)


"Thread-3" Id=14 cpuUsage=84.42% deltaTime=178ms time=17315ms RUNNABLE
    at learn.jvm.CPU.lambda$main$0(CPU.java:13)
    at learn.jvm.CPU$$Lambda$1/500977346.run(Unknown Source)
    at java.lang.Thread.run(Thread.java:748)


  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

3.3 非堆空间导致OOM

3.3.1 模拟异常

启动参数: -XX:MetaspaceSize=120m -XX:MaxMetaspaceSize=120m -XX:+PrintGCDetails -XX:+PrintGCDateStamps -Xloggc:/Users/liuxin/Github/learn-example/logs/gc.log

public class MetaspaceOverflowTest {

    /**
     * 查看元空间配置
     * java -XX:+PrintFlagsFinal -version | grep Metaspace
     * 方法区是JVM规范。
     * - 永久代和元空间是实现
     * 元空间调优规则:
     * 1. 最大最小设置成一样大
     * 防止内存抖动
     *
     * @param args -XX:MetaspaceSize=20m
     *             -XX:MaxMetaspaceSize=20m
     *             java.lang.OutOfMemoryError-->Metaspace
     */
    public static void main(String[] args) {
//        while (true) {
            Sleeps.sleep(0.2);
            for (int i = 0; i < 200; i++) {
                Enhancer enhancer = new Enhancer();
                enhancer.setSuperclass(MetaspaceOverflowTest.class);
                enhancer.setUseCache(false);
                enhancer.setCallback(new MethodInterceptor() {
                    @Override
                    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                        return methodProxy.invokeSuper(objects, args);
                    }
                });
                System.out.println("create InstanceKlass...");
                enhancer.create();
            }
//        }
        while (true);
        //  java.lang.OutOfMemoryError-->Metaspace
    }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36

3.3.2 现象

当你收到运维告警,或者是明显感觉到系统吞吐量下降,甚至会有oom异常的时候,首先先去看下 GC日志,找到GC的原因。下面看下非堆空间溢出导致的GC日志,并配上前面的GC日志学习。来
排查下问题。

CommandLine flags: -XX:CompressedClassSpaceSize=12582912 -XX:InitialHeapSize=268435456 -XX:MaxHeapSize=4294967296 -XX:MaxMetaspaceSize=20971520 -XX:MetaspaceSize=20971520 -XX:+PrintGC -XX:+PrintGCDateStamps -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseParallelGC 
2022-06-20T18:12:26.411-0800: 1.749: [GC (Allocation Failure) [PSYoungGen: 65536K->3006K(76288K)] 65536K->3006K(251392K), 0.0233877 secs] [Times: user=0.06 sys=0.02, real=0.02 secs] 
2022-06-20T18:12:26.960-0800: 2.298: [GC (Allocation Failure) [PSYoungGen: 68542K->3264K(141824K)] 68542K->3272K(316928K), 0.0194187 secs] [Times: user=0.08 sys=0.02, real=0.02 secs] 
2022-06-20T18:12:27.824-0800: 3.162: `[GC (Allocation Failure)` [PSYoungGen: 134336K->5264K(141824K)] 134344K->5280K(316928K), 0.0145565 secs] [Times: user=0.06 sys=0.02, real=0.01 secs] 
2022-06-20T18:12:28.526-0800: 3.864: [GC (Allocation Failure) [PSYoungGen: 136336K->6928K(272896K)] 136352K->6952K(448000K), 0.0198281 secs] [Times: user=0.09 sys=0.03, real=0.02 secs] 
2022-06-20T18:12:29.252-0800: 4.590: [GC (Metadata GC Threshold) [PSYoungGen: 187304K->8848K(272896K)] 187328K->8880K(448000K), 0.0217320 secs] [Times: user=0.10 sys=0.02, real=0.02 secs] 
2022-06-20T18:12:29.274-0800: 4.612: [Full GC (Metadata GC Threshold) [PSYoungGen: 8848K->0K(272896K)] [ParOldGen: 32K->8685K(86016K)] 8880K->8685K(358912K), [Metaspace: 20088K->20088K(1069056K)], 0.0245986 secs] [Times: user=0.10 sys=0.01, real=0.02 secs] 
2022-06-20T18:12:29.299-0800: 4.637: [GC (Last ditch collection) [PSYoungGen: 0K->0K(476160K)] 8685K->8685K(562176K), 0.0005319 secs] [Times: user=0.01 sys=0.00, real=0.01 secs] 
2022-06-20T18:12:29.300-0800: 4.638: [Full GC (Last ditch collection) [PSYoungGen: 0K->0K(476160K)] [ParOldGen: 8685K->3731K(155648K)] 8685K->3731K(631808K), [Metaspace: 20088K->20088K(1069056K)], 0.0187273 secs] [Times: user=0.07 sys=0.01, real=0.01 secs] 

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

从面的GC日志中我们能找到些GC原因,通过前面的学习。我们可以判断出来。

属于非堆空间造成的OOM。

3.3.3 解决方案

  1. 排查看应用中是否有动态创建Class的地方
  2. 添加元空间大小(如果应用配置限制元空间大小,还出现了这样的问题,一般一定是程序中有bug导致)

文章来源: springlearn.blog.csdn.net,作者:西魏陶渊明,版权归原作者所有,如需转载,请联系作者。

原文链接:springlearn.blog.csdn.net/article/details/126493214

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。