JVM内存、CPU占用过高问题定位


前言

如果开发过程中代码存在不规范漏洞,Java程序在运行时就很有可能会出现:内存(溢出、泄漏)、CPU占用过高。本篇文章将通过一个简单的例子给大家演示一下,如果线上出现以上问题我们可以如何精准定位到代码上的问题。

排查思路:进程->线程->代码->类->方法


一、示例代码

模拟一个线程往一个集合List<byte[]> list中无限循环添加byte[],这样就可以有效的观察到java进程运行时会出现CPU、内存占用过高的场景。然后给当前运行的线程自定一个线程名称,便于后续问题排查。

package com.nb.java;

import java.util.ArrayList;
import java.util.List;

@SpringBootApplication
public class JavaBasisApplication {
	public static void main(String[] args) {
		// 定义一个list
		List<byte[]> list = new ArrayList<>();
        // 创建一个线程,并且自定义这个线程的名称:模拟Java运行时CPU、内存占用过高的线程
		new Thread(() -> {
			while (true){ 
				System.out.println("===每次添加一个byte[10]字节数组===");
				list.add(new byte[10]);
			}
		},"模拟Java运行时CPU、内存占用过高的线程").start();
	}
}

二、运行代码

1、将上面的类基于SpringBoot打包成一个jar包,随便放到一个服务器运行。执行以下命令:

nohup java -jar java.jar > test.log 2>&1 &

2、查询当前运行的java程序进程PID:882245

jps -l

在这里插入图片描述

三、top命令

在这里插入图片描述

  • CPU=262.4%
  • MEM=5.3%

通过top命令可以发现882245这个进程(也就是我们运行的这个java程序)占用的CPU和内存相对来说都是比较高的。

四、问题定位

1、查看java进程所有的线程列表

top -p 882245 -H

在这里插入图片描述

从这个图可以看出我们运行的java程序里出现了882292线程占用CPU、内存过高。到这里我们就成功了一半。至少看到了是那个线程出现的问题。

2、输出线程882292十六进制

printf "%x\n" 882292

d7674(882292的十六进制的值,用于后面查看线程堆栈中的关键字检索)

五、输出java堆栈信息

注意:这里的882245是进程ID,不是线程ID。

jstack 882245 > pid_882245_dump.txt 

在当前目录会生成一个pid_882245_dump.txt 文件。
在这里插入图片描述

六、pid_882245_dump.txt

cat pid_882245_dump.txt | grep d7674 -B 30 

这里的(d7674)就是第四部生成的。

root@yiliao:~# cat pid_882245_dump.txt | grep d7674 -A 30 
"模拟Java运行时CPU、内存占用过高的线程" #31 prio=5 os_prio=0 cpu=1117037.88ms elapsed=1168.34s tid=0x00007efd81e31000 nid=0xd7674 runnable  [0x00007efd1c327000]
   java.lang.Thread.State: RUNNABLE
	at java.io.FileOutputStream.writeBytes(java.base@11.0.19/Native Method)
	at java.io.FileOutputStream.write(java.base@11.0.19/FileOutputStream.java:354)
	at java.io.BufferedOutputStream.flushBuffer(java.base@11.0.19/BufferedOutputStream.java:81)
	at java.io.BufferedOutputStream.flush(java.base@11.0.19/BufferedOutputStream.java:142)
	- locked <0x0000000413bcfa60> (a java.io.BufferedOutputStream)
	at java.io.PrintStream.write(java.base@11.0.19/PrintStream.java:561)
	- locked <0x00000004119f6828> (a java.io.PrintStream)
	at sun.nio.cs.StreamEncoder.writeBytes(java.base@11.0.19/StreamEncoder.java:233)
	at sun.nio.cs.StreamEncoder.implFlushBuffer(java.base@11.0.19/StreamEncoder.java:312)
	at sun.nio.cs.StreamEncoder.flushBuffer(java.base@11.0.19/StreamEncoder.java:104)
	- locked <0x00000004119f67e0> (a java.io.OutputStreamWriter)
	at java.io.OutputStreamWriter.flushBuffer(java.base@11.0.19/OutputStreamWriter.java:181)
	at java.io.PrintStream.write(java.base@11.0.19/PrintStream.java:606)
	- eliminated <0x00000004119f6828> (a java.io.PrintStream)
	at java.io.PrintStream.print(java.base@11.0.19/PrintStream.java:745)
	at java.io.PrintStream.println(java.base@11.0.19/PrintStream.java:882)
	- locked <0x00000004119f6828> (a java.io.PrintStream)
	at com.nb.java.JavaBasisApplication.lambda$main$0(JavaBasisApplication.java:30)
	at com.nb.java.JavaBasisApplication$$Lambda$634/0x00000008403d2c40.run(Unknown Source)
	at java.lang.Thread.run(java.base@11.0.19/Thread.java:829)

"DestroyJavaVM" #32 prio=5 os_prio=0 cpu=2896.41ms elapsed=1168.34s tid=0x00007efd80015000 nid=0xd7646 waiting on condition  [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Attach Listener" #33 daemon prio=9 os_prio=0 cpu=1.12ms elapsed=18.83s tid=0x00007efd18001000 nid=0xd76d7 waiting on condition  [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"VM Thread" os_prio=0 cpu=781.96ms elapsed=1171.43s tid=0x00007efd802a6000 nid=0xd764c runnable

在这里插入图片描述

从图上下面的剪头指向就很写清楚定位到:com.nb.java.JavaBasisApplication#main方法有问题。这样就定位到我们的具体代码。

  • 19
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 调优 JVM 的方法有很多,常用的方法如下: 1. 调整堆内存大小。可以使用 `-Xms` 和 `-Xmx` 参数来调整堆内存的初始大小和最大大小。如果堆内存过小,会导致频繁的 GC,从而导致 CPU 占用过高。 2. 调整 GC 策略。JVM 中有很多不同的 GC 算法,如 Serial GC、Parallel GC、CMS GC 等。每种算法都有自己的优缺点。可以使用 `-XX:+UseSerialGC` 或者 `-XX:+UseParallelGC` 等参数来选择不同的 GC 算法。 3. 调整 JVM 参数。JVM 还有很多其他的参数,比如 `-XX:+PrintGC` 可以在控制台输出 GC 的日志,`-XX:+PrintGCDetails` 可以输出详细的 GC 日志,方便我们分析问题。 4. 使用 profiler 分析 CPU 占用情况。有很多 profiler 工具可以帮助我们分析 CPU 占用情况。比如 Java Flight Recorder、JProfiler 等。使用这些工具可以帮助我们找到 CPU 占用过高的原因,从而调优 JVM。 ### 回答2: 要优化JVM来降低CPU占用过高问题,可以从以下几个方面进行优化: 1. 调整内存设置:通过适当调整JVM内存设置来避免内存压力过大,导致频繁的垃圾回收。可以通过修改-Xms和-Xmx参数来调整JVM的初始堆和最大堆大小,需要根据实际情况进行调整。 2. 减少对象创建:过多的对象创建会导致频繁的垃圾回收,从而造成CPU占用过高。可以通过优化代码,避免不必要的对象创建,使用对象池等方式减少对象的产生。 3. 优化算法和数据结构:选择合适的算法和数据结构可以减少CPU的使用率。对于一些耗时的操作,可以考虑使用更高效的算法来替代。 4. 多线程优化:合理的使用多线程可以充分利用多核CPU的性能,减少单个线程的CPU占用率。可以将一些独立、并行的任务放入不同的线程中进行处理。 5. 使用性能分析工具:可以使用性能分析工具来检测和分析CPU占用过高的原因,如JProfiler、VisualVM等。通过定位到具体的问题,可以有针对性地进行优化。 总之,JVM的优化需要综合考虑内存设置、对象创建、算法数据结构、多线程等多个因素。通过适当的调整和优化,可以降低CPU占用过高问题,提高系统的性能。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值