Linux下如何定位最耗CPU的JAVA代码

简介: Linux下如何定位最耗CPU的JAVA代码

引言

程序运行卡的时候,我们经常需要定位到底程序在执行什么代码最消耗CPU,这里有个我经常用的小技巧,分享一下。

过程如下:

首先,Linux下面top命令可以直接找出最消耗CPU的进程  
top

可以看到如下显示,我们看到前面有三个都是100% CPU占用的

top - 19:53:23 up 118 days,  6:09,  4 users,  load average: 3.00, 2.68, 1.68
Tasks: 446 total,   1 running, 438 sleeping,   7 stopped,   0 zombie
Cpu(s):  8.6%us,  4.2%sy,  0.0%ni, 87.2%id,  0.0%wa,  0.0%hi,  0.0%si,  0.0%st
Mem:  132103700k total, 76585104k used, 55518596k free,   424052k buffers
Swap: 16383996k total,        0k used, 16383996k free, 36290728k cached

   PID USER      PR  NI  VIRT  RES  SHR S %CPU %MEM    TIME+  COMMAND                                                                                                                                             
156888 hdfs      20   0 34.9g 544m  11m S 100.8  0.4   9:44.08 java                                                                                                                                                
156928 hdfs      20   0 34.9g 545m  11m S 100.8  0.4   9:42.84 java                                                                                                                                                
156769 hdfs      20   0 34.9g 544m  11m S 100.2  0.4   9:53.26 java                                                                                                                                                
153980 hdfs      20   0 64.0g  24g  55m S  1.7 19.6   5879:56 presto-server                                                                                                                                        
   105 root      20   0     0    0    0 S  0.3  0.0  45:26.08 events/6                                                                                                                                             
   106 root      20   0     0    0    0 S  0.3  0.0   4:41.29 events/7   代码片

接下来,我们一个个分析,先找第一个最消耗CPU的,我们使用命令

top -Hp 156888

这个命令可以展示进程内最消耗CPU的线程,可以得到如下结果

top - 20:00:25 up 118 days,  6:16,  2 users,  load average: 3.00, 2.91, 2.14
Tasks:  39 total,   1 running,  38 sleeping,   0 stopped,   0 zombie
Cpu(s):  9.3%us,  3.5%sy,  0.0%ni, 87.2%id,  0.0%wa,  0.0%hi,  0.0%si,  0.0%st
Mem:  132103700k total, 76578608k used, 55525092k free,   424052k buffers
Swap: 16383996k total,        0k used, 16383996k free, 36290948k cached

   PID USER      PR  NI  VIRT  RES  SHR S %CPU %MEM    TIME+  COMMAND                                                                                                                                             
156926 hdfs      20   0 34.9g 544m  11m R 99.8  0.4  16:43.39 java                                                                                                                                                 
156901 hdfs      20   0 34.9g 544m  11m S  0.3  0.4   0:00.16 java                                                                                                                                                 
156888 hdfs      20   0 34.9g 544m  11m S  0.0  0.4   0:00.00 java                                                                                                                                                 
156889 hdfs      20   0 34.9g 544m  11m S  0.0  0.4   0:00.12 java                                                                                                                                                 
156890 hdfs      20   0 34.9g 544m  11m S  0.0  0.4   0:00.16 java                                                                                                                                                 
156891 hdfs      20   0 34.9g 544m  11m S  0.0  0.4   0:00.17 java                                                                                                                                                 
156892 hdfs      20   0 34.9g 544m  11m S  0.0  0.4   0:00.18 java                                                                                                                                                 
156893 hdfs      20   0 34.9g 544m  11m S  0.0  0.4   0:00.17 java

排名第一的便是我们这个进程下面的线程号了。

接下来,我们使用一个java中的jstack命令去定位哪一行:

jstack 156888

这样子直接获取信息内容是这样子的:

"Service Thread" #17 daemon prio=9 os_prio=0 tid=0x00002b47080f6800 nid=0x264fc runnable [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C1 CompilerThread11" #16 daemon prio=9 os_prio=0 tid=0x00002b47080f3800 nid=0x264fb waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C1 CompilerThread10" #15 daemon prio=9 os_prio=0 tid=0x00002b47080f1000 nid=0x264fa waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C1 CompilerThread9" #14 daemon prio=9 os_prio=0 tid=0x00002b47080ef000 nid=0x264f9 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C1 CompilerThread8" #13 daemon prio=9 os_prio=0 tid=0x00002b47080ed000 nid=0x264f8 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

我们大致读得出这个线程号,但是我们需要定位最消耗cpu的那个线程,我们看到nid项其实就是线程号,不过是16进制输出的,所以我们也要把我们的线程号转换成16进制,可以直接用计算器,没问题,我这里用linux下面打印命令:

printf "%x\n" 156926

结果为:

264fe

这个时候我们可以用这个数字结合我们的jstack命令去查询了:

jstack 156888 |grep 264fe

我们这样子可以搜索到这个线程,但是只有一行结果

"thread1" #18 prio=5 os_prio=0 tid=0x00002b4708167000 nid=0x264fe runnable [0x00002b4723545000]

我们可以在grep前后增加要显示的行数,可以看到上下文的内容,完整命令如下:

jstack 156888 |grep -A 20 -B 10 264fe 

我们可以输出我们要的结果

"thread1" #18 prio=5 os_prio=0 tid=0x00002b4708167000 nid=0x264fe runnable [0x00002b4723545000]
   java.lang.Thread.State: RUNNABLE
        at java.io.FileOutputStream.writeBytes(Native Method)
        at java.io.FileOutputStream.write(FileOutputStream.java:326)
        at java.io.BufferedOutputStream.flushBuffer(BufferedOutputStream.java:82)
        at java.io.BufferedOutputStream.flush(BufferedOutputStream.java:140)
        - locked <0x0000000080208ea8> (a java.io.BufferedOutputStream)
        at java.io.PrintStream.write(PrintStream.java:482)
        - locked <0x00000000802048c0> (a java.io.PrintStream)
        at sun.nio.cs.StreamEncoder.writeBytes(StreamEncoder.java:221)
        at sun.nio.cs.StreamEncoder.implFlushBuffer(StreamEncoder.java:291)
        at sun.nio.cs.StreamEncoder.flushBuffer(StreamEncoder.java:104)
        - locked <0x00000000802048a8> (a java.io.OutputStreamWriter)
        at java.io.OutputStreamWriter.flushBuffer(OutputStreamWriter.java:185)
        at java.io.PrintStream.write(PrintStream.java:527)
        - eliminated <0x00000000802048c0> (a java.io.PrintStream)
        at java.io.PrintStream.print(PrintStream.java:669)
        at java.io.PrintStream.println(PrintStream.java:806)
        - locked <0x00000000802048c0> (a java.io.PrintStream)
        at Demo1_16.lambda$main$0(Demo1_16.java:6)
        at Demo1_16$$Lambda$1/834600351.run(Unknown Source)

从内容上面看,我们尅看到是在第6行代码上面有个打印,我把完整代码贴出来:

public class Demo1_16 {
    public static void main(String[] args) {
       new Thread(()->{
           while (true){
               System.out.println("hi!");
           }
       },"thread1").start();
    }
}

可以看到是有个死循环导致的,至于为什么有三个占用最高的,那是因为我同时启动了三个这个程序。

我们把命令可以直接连起来:

jstack 156888 |grep `printf "%x" 264fe` -C6 --color

总结:

要定位最消耗的代码,我们的思路是缩小范围,顺序是进程->线程->代码,最后再看实际代码去解决问题

目录
打赏
0
0
0
0
2
分享
相关文章
接替此文【下篇-服务端+后台管理】优雅草蜻蜓z系统JAVA版暗影版为例-【蜻蜓z系列通用】-2025年全新项目整合搭建方式-这是独立吃透代码以后首次改变-独立PC版本vue版搭建教程-优雅草卓伊凡
接替此文【下篇-服务端+后台管理】优雅草蜻蜓z系统JAVA版暗影版为例-【蜻蜓z系列通用】-2025年全新项目整合搭建方式-这是独立吃透代码以后首次改变-独立PC版本vue版搭建教程-优雅草卓伊凡
181 96
接替此文【下篇-服务端+后台管理】优雅草蜻蜓z系统JAVA版暗影版为例-【蜻蜓z系列通用】-2025年全新项目整合搭建方式-这是独立吃透代码以后首次改变-独立PC版本vue版搭建教程-优雅草卓伊凡
|
8天前
|
Java代码结构解析:类、方法、主函数(1分钟解剖室)
### Java代码结构简介 掌握Java代码结构如同拥有程序世界的建筑蓝图,类、方法和主函数构成“黄金三角”。类是独立的容器,承载成员变量和方法;方法实现特定功能,参数控制输入环境;主函数是程序入口。常见错误包括类名与文件名不匹配、忘记static修饰符和花括号未闭合。通过实战案例学习电商系统、游戏角色控制和物联网设备监控,理解类的作用、方法类型和主函数任务,避免典型错误,逐步提升编程能力。 **脑图速记法**:类如太空站,方法即舱段;main是发射台,static不能换;文件名对仗,括号要成双;参数是坐标,void不返航。
28 5
Linux系统查看操作系统版本信息、CPU信息、模块信息
在Linux系统中,常用命令可帮助用户查看操作系统版本、CPU信息和模块信息
160 23
如何在 Java 代码中使用 JSqlParser 解析复杂的 SQL 语句?
大家好,我是 V 哥。JSqlParser 是一个用于解析 SQL 语句的 Java 库,可将 SQL 解析为 Java 对象树,支持多种 SQL 类型(如 `SELECT`、`INSERT` 等)。它适用于 SQL 分析、修改、生成和验证等场景。通过 Maven 或 Gradle 安装后,可以方便地在 Java 代码中使用。
423 11
利用 Java 代码获取淘宝关键字 API 接口
在数字化商业时代,精准把握市场动态与消费者需求是企业成功的关键。淘宝作为中国最大的电商平台之一,其海量数据中蕴含丰富的商业洞察。本文介绍如何通过Java代码高效、合规地获取淘宝关键字API接口数据,帮助商家优化产品布局、制定营销策略。主要内容包括: 1. **淘宝关键字API的价值**:洞察用户需求、优化产品标题与详情、制定营销策略。 2. **获取API接口的步骤**:注册账号、申请权限、搭建Java开发环境、编写调用代码、解析响应数据。 3. **注意事项**:遵守法律法规与平台规则,处理API调用限制。 通过这些步骤,商家可以在激烈的市场竞争中脱颖而出。
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
90 3
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
90 2
JavaEE初阶——初识EE(Java诞生背景,CPU详解)
带你从零入门JAVAEE初阶,Java的发展历程认识什么是cpu,cpu的工作原理,cpu是如何进行计算的,cpu的架构,指令集,cpu的核心,如何提升cpu的算力,cpu的指令,,cup的缓存,cpu的流水线
04. 【Java教程】在 Linux 上安装 Java
04. 【Java教程】在 Linux 上安装 Java
520 0

热门文章

最新文章