Java应用CPU飙升和死锁排查实战教程

news2025/4/13 8:17:48

引言

在日常开发中,我们可能会遇到Java应用CPU飙升和死锁的问题。本文将通过实际案例,为大家介绍如何排查这些问题

Java应用CPU飙升和死锁排查步骤

  1. 先执行top命令,找到CPU占用比较高的进程
  2. 再执行jstack 进程id > dump.txt
  3. 找到进程中CPU占用比较高的线程,线程id转16进制
  4. 到dump.txt文件中根据线程id查看线程的具体状态即可

下面我们使用简单的例子检验排查步骤

首先我们先写两个Demo模拟CPU飙升和死锁的情况,启动SpringBoot项目发送Http请求触发这个两个接口,其中loop接口触发了两次,达到更好演示CPU飙升的效果。

package your.controller;

import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.concurrent.TimeUnit;

@RestController
@Slf4j
public class CpuController {

    private Object lock1 = new Object();
    private Object lock2 = new Object();


    @GetMapping("loop")
    public String loop() {
        log.warn("loop start");
        while (true) {}
    }

    @GetMapping("deadlock")
    public String deadlock() {
        log.warn("deadlock start");
        new Thread(()->{
            synchronized (lock1) {
                try {
                    TimeUnit.SECONDS.sleep(1);
                } catch (InterruptedException e) {}
                synchronized (lock2) {
                    log.info("thread1 over");
                }
            }
        }).start();

        new Thread(()->{
            synchronized (lock2) {
                try {
                    TimeUnit.SECONDS.sleep(1);
                } catch (InterruptedException e) {}
                synchronized (lock1) {
                    log.info("thread2 over");
                }
            }
        }).start();

        return "ok";
    }


}

步骤一:先执行top命令,找到CPU占用比较高的进程

在这里插入图片描述
可以看到进程号为 279738 的应用占用CPU较高。

步骤二: 再执行jstack 进程id > dump.txt生成线程快照

在此例子中执行:jstack 279738 > dump.txt
在这里插入图片描述

步骤三: 找到进程中CPU占用比较高的线程,线程id转16进制

在此例子中执行:top -p 279738 -H,可以看到279758279761这两个线程占用CPU较高(由于之前触发了两次loop接口,所有这两个线程占用较高)
在这里插入图片描述
因为dump文件的线程ID是使用16进制表示的,所以再执行:printf "%x" 279758,将10进制转成16进制,是 444ce
在这里插入图片描述

步骤四: 到dump.txt文件中根据线程id查看线程的具体状态即可

在这里插入图片描述
可以看出线程状态是RUNNABLE,并且运行到了20行

at com.example.demo.controller.CpuController.loop(CpuController.java:20)

回到代码分析,确实是在第20行发生了死循环。
在这里插入图片描述


在dump文件的末尾可以看到死锁的情况
在这里插入图片描述


Found one Java-level deadlock:
=============================
"Thread-8":
  waiting to lock monitor 0x00007f71340066d8 (object 0x00000000e2eba7e0, a java.lang.Object),
  which is held by "Thread-6"
"Thread-6":
  waiting to lock monitor 0x00007f714001fad8 (object 0x00000000e2eba7d0, a java.lang.Object),
  which is held by "Thread-5"
"Thread-5":
  waiting to lock monitor 0x00007f71340066d8 (object 0x00000000e2eba7e0, a java.lang.Object),
  which is held by "Thread-6"

Java stack information for the threads listed above:
===================================================
"Thread-8":
	at com.example.demo.controller.CpuController.lambda$deadlock$1(CpuController.java:40)
	- waiting to lock <0x00000000e2eba7e0> (a java.lang.Object)
	at com.example.demo.controller.CpuController$$Lambda$475/725445379.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)
"Thread-6":
	at com.example.demo.controller.CpuController.lambda$deadlock$1(CpuController.java:43)
	- waiting to lock <0x00000000e2eba7d0> (a java.lang.Object)
	- locked <0x00000000e2eba7e0> (a java.lang.Object)
	at com.example.demo.controller.CpuController$$Lambda$475/725445379.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)
"Thread-5":
	at com.example.demo.controller.CpuController.lambda$deadlock$0(CpuController.java:32)
	- waiting to lock <0x00000000e2eba7e0> (a java.lang.Object)
	- locked <0x00000000e2eba7d0> (a java.lang.Object)
	at com.example.demo.controller.CpuController$$Lambda$474/637174220.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)

Found 1 deadlock.

可以看出Thread-8想要得到Thread-6的资源,Thread-6想要得到Thread-5的资源,Thread-5想要得到Thread-6的资源,回到代码可以看出死锁的具体位置。
在这里插入图片描述
按照上述排查步骤,我们成功地找到了导致CPU飙升和死锁的原因。在实际操作中,我们可以根据这些信息对代码进行优化,解决性能问题。

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

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

相关文章

软考127-上午题-【软件工程】-McCabe度量法

一、McCabe度量法 1-1、定义 McCabe 度量法是通过定义环路复杂度&#xff0c;建立程序复杂性的度量。 它基于一个程序模块的程序图中环路的个数。计算有向图G的环路复杂性的公式为&#xff1a; V(G) m - n 2 闭合区域 1 其中V(G)是有向图 G 中的环路个数&#xff0c;m 是…

[全网最全]2024MathorCup妈妈杯ABCD题成品论文33页+配套完整代码数据汇总

所有题目的每一小问解答&#xff08;含配套代码和数据&#xff09;都已经更新完毕&#xff0c;其中C题成品论文33页更新&#xff0c;B题论文更新&#xff0c;A题半成品论文21页完整解答代码数据。 &#xff08;完整版的资料放在文末了&#xff09; A题 移动通信网络中PCI规划问…

蓝桥杯嵌入式(G431)备赛笔记——第十一届第二场真题

关键代码&#xff1a;、 user.c: u32 adc_tick 0; // 定义一个无符号32位整型变量 adc_tick&#xff0c;用于记录上次ADC处理的时间戳 u32 r37_value 0; // 定义一个无符号32位整型变量 r37_value&#xff0c;用于存储ADC通道2的采样值 u32 r38_value 0; // 定义一个无符号…

AI常见关键术语

哈喽&#xff0c;大家好&#xff0c;我是小码哥&#xff0c;人工智能技术的快速发展带来了许多专业术语&#xff0c;这些词汇对于理解AI的工作原理和应用至关重要。以下是一些关键的AI术语&#xff0c;以及它们的专业解释和通俗总结。 一、核心概念 人工智能 (AI) 专业解释&am…

轻量带屏解决方案之恒玄芯片移植案例

本文章基于恒玄科技BES2600W芯片的欧智通 Multi-modal V200Z-R开发板 &#xff0c;进行轻量带屏开发板的标准移植&#xff0c;开发了智能开关面板样例&#xff0c;同时实现了ace_engine_lite、arkui_ui_lite、aafwk_lite、appexecfwk_lite、HDF等部件基于OpenHarmony LiteOS-M内…

AI预测体彩排3第3弹【2024年4月14日预测--第1套算法开始计算第3次测试】

今天咱们继续测试第1套算法和模型&#xff0c;今天是第3次测试&#xff0c;目前的测试只是为了记录和验证&#xff0c;不建议大家盲目跟买。我的目标仍旧是10次命中3-4次!~废话不多说了&#xff0c;直接上结果&#xff01; 2024年4月14日排3的七码预测结果如下 第一套&…

mybatis的一对多

业务&#xff1a;通常主表从表 查询&#xff0c;一对多关系&#xff0c;通常是先查主表&#xff0c;然后拿主表的 关联字段与从表关联。在代码中 通常用for 循环等方法给 从表的数据赋值&#xff0c;很麻烦&#xff0c;&#xff0c;&#xff0c;很麻烦。。。。 用mybatis的…

软考中级--网络工程师-计算机基础与理论第二节无线基础知识

IEEE802.11 规定了多种 WLAN 通信标准&#xff0c;其中&#xff08; &#xff09;与其他标准采用的频段不同&#xff0c;因而不能兼容。 A IEEE802.11a B IEEE802.11b C IEEE802.11g D IEEE802.11n 试题答案 正确答案&#xff1a; A 答案解析 IEEE 802.11a规定采用5GHz的 ISM频…

Python | Leetcode Python题解之第25题K个一组翻转链表

题目&#xff1a; 题解&#xff1a; class Solution:# 翻转一个子链表&#xff0c;并且返回新的头与尾def reverse(self, head: ListNode, tail: ListNode):prev tail.nextp headwhile prev ! tail:nex p.nextp.next prevprev pp nexreturn tail, headdef reverseKGroup…

关于时频分析的一些事-答知乎问(一)

从信号的时频谱图中可以提取什么特征&#xff1f; 基于时频谱图的特征一般包括能量特征、时域和频域拓展特征以及时频内禀特征。 基于时频图的能量特征 基于时频图的特征中&#xff0c;能量特征是最简单的一种&#xff0c;通过分析时频谱图中的能量分布特性而获取信号的时频…

振兴国腾GM8775C MIPIDSI桥接到双 PORT LVDS

GM8775C描述&#xff1a; GM8775C 型 DSI 转双通道 LVDS 发送器产品主要实现将 MIPI DSI 转单 / 双通道 LVDS 功能&#xff0c;MIPI 支持 1/2/3/4 通道可选&#xff0c;每通道最高支持 1Gbps 速率&#xff0c;最大支持 4Gbps 速率。LVDS 时钟频率高达 154MHz &a…

SqlServer专题

目录 1&#xff0c;连接数据库 2&#xff0c;连接池 1.何为连接池&#xff1f; 2.连接池运行原理。 3.如何查看连接池&#xff1f; 4.连接池注意事项。 3&#xff0c;一般SQL语句。 4&#xff0c;控制语句 1.判断语句 2.循环语句 5&#xff0c;视图 1.使用…

vue-router 原理【详解】hash模式 vs H5 history 模式

hash 模式 【推荐】 路由效果 在不刷新页面的前提下&#xff0c;根据 URL 中的 hash 值&#xff0c;渲染对应的页面 http://test.com/#/login 登录页http://test.com/#/index 首页 核心API – window.onhashchange 监听 hash 的变化&#xff0c;触发视图更新 window.onhas…

黑马 javaweb 实现小案例

黑马 javaweb 实现案例 环境搭建 配置文件代码&#xff1a; spring:datasource:driver-class-name: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://localhost:3306/demo0413username: rootpassword: 123456mybatis:configuration:#配置mybatis的日志, 指定输出到控制台log-impl…

Java-博客系统(前后端交互)

目录 前言 博客系统基本情况 1 创建项目&#xff0c;引入依赖 2 数据库设计 2.1 分析 2.2 建库建表 3 封装数据库 3.1 在java目录下创建DBUtil类&#xff0c;通过这个类对数据库进行封装 3.2 在java目录下创建实体类&#xff08;博客类Blog&#xff09; 3.2 在java目录下创建…

Python里安装了库却报错找不到是怎么回事?

你在写代码的时候有没有遇到过这样的问题&#xff1a; 明明已经用pip安装好了一个Python模块&#xff0c; 但当你在代码中使用时&#xff0c;却给你报错说找不到这个库。 出现这种情况&#xff0c;绝大多数都是因为你安装模块的那个pip&#xff0c;和你执行代码时的python&…

机器学习—特征预处理和降维(四)

什么是特征预处理&#xff1f; 通过一些转换函数将特征数据转换成更加适合算法模型的特征数据过程 1包含内容 数值型数据的无量纲化&#xff1a; 归一化标准化 2特征预处理API sklearn. preprocessing为什么要进行归一化 or 标准化&#xff1f; 特征的单位或者大小相差较大…

UDP实现Mini版在线聊天室

实现原理 只有当客户端先对服务器发送online消息的时候&#xff0c;服务器才会把客户端加入到在线列表。当在线列表的用户发消息的时候&#xff0c;服务器会把消息广播给在线列表中的所有用户。而当用户输入offline时&#xff0c;表明自己要下线了&#xff0c;此时服务器把该用…

keil参数删除后补写没有代码提示

代码提示快捷键 ctrlalt空格 如果按了之后没有提示&#xff0c;那说明跟输入法的快捷键冲突了。

C++引用和右值引用

我最近开了几个专栏&#xff0c;诚信互三&#xff01; > |||《算法专栏》&#xff1a;&#xff1a;刷题教程来自网站《代码随想录》。||| > |||《C专栏》&#xff1a;&#xff1a;记录我学习C的经历&#xff0c;看完你一定会有收获。||| > |||《Linux专栏》&#xff1…