一文带你深入解析Java应用线程转储

 更新时间:2025年04月23日 10:46:59   作者:码农阿豪@新空间  
在Java应用运维和问题排查过程中,线程转储(Thread Dump)是一个非常重要的工具,它能够帮助我们了解JVM内部线程的运行状态,下面小编来和大家详细介绍一下它吧

引言

在Java应用运维和问题排查过程中,线程转储(Thread Dump)是一个非常重要的工具,它能够帮助我们了解JVM内部线程的运行状态,快速定位死锁、线程阻塞、资源竞争等问题。本文将通过一个实际的线程转储日志案例,详细分析其内容,并结合代码示例,讲解如何从中发现问题并优化应用性能。

1. 什么是线程转储

线程转储是JVM在某一时刻所有线程的快照,包含每个线程的调用栈、状态和锁信息。通过分析线程转储,我们可以:

  • 检查线程是否阻塞或死锁。
  • 发现资源竞争问题。
  • 优化线程池配置。
  • 诊断应用响应缓慢或崩溃的原因。

如何获取线程转储?

使用 jstack 命令(适用于运行中的Java进程):

jstack -l <pid> > thread_dump.log

通过 kill -3 发送信号(适用于Linux环境):

kill -3 <pid>

使用JMX工具(如VisualVM、JConsole)。

2. 案例日志分析

我们分析的日志片段如下:

2025-04-22 18:16:40
Full thread dump OpenJDK 64-Bit Server VM (25.362-b08 mixed mode):

"SIGTERM handler" #138 daemon prio=9 os_prio=0 tid=0x00007f03fc005000 nid=0xa9b06 runnable [0x00007f0438dfc000]
   java.lang.Thread.State: RUNNABLE
    at java.lang.Thread.run(Thread.java:749)

"Druid-ConnectionPool-Destroy-816944408" #137 daemon prio=5 os_prio=0 tid=0x00007f0364222000 nid=0xa9aff waiting on condition [0x00007f04385fc000]
   java.lang.Thread.State: TIMED_WAITING (sleeping)
    at java.lang.Thread.sleep(Native Method)
    at com.alibaba.druid.pool.DruidDataSource$DestroyConnectionThread.run(DruidDataSource.java:2786)
...

关键线程解析

(1) SIGTERM handler 线程

状态:RUNNABLE

作用:处理JVM终止信号(如 kill -15),表明应用正在关闭。

可能的问题:如果是非预期关闭,需检查是否有异常终止或OOM。

(2) Druid 连接池线程

Druid-ConnectionPool-Destroy-*

  • 状态:TIMED_WAITING (sleeping)
  • 作用:销毁闲置数据库连接。
  • 优化建议:调整 timeBetweenEvictionRunsMillis 参数,避免频繁销毁。

Druid-ConnectionPool-Create-*

  • 状态:WAITING (parking)
  • 作用:创建新数据库连接。
  • 潜在问题:如果长期阻塞,可能是连接池耗尽,需检查 maxActive 配置。

(3) Nacos 客户端线程

nacos-grpc-client-executor-*

状态:TIMED_WAITING (parking)

作用:Nacos 客户端通过gRPC与服务端通信。

排查点:如果大量线程阻塞,可能是Nacos服务端不可达或网络问题。

3. 常见问题及解决方案

3.1 线程死锁

示例代码:

public class DeadlockExample {
    private static final Object lock1 = new Object();
    private static final Object lock2 = new Object();

    public static void main(String[] args) {
        new Thread(() -> {
            synchronized (lock1) {
                try { Thread.sleep(100); } catch (InterruptedException e) {}
                synchronized (lock2) {
                    System.out.println("Thread 1");
                }
            }
        }).start();

        new Thread(() -> {
            synchronized (lock2) {
                synchronized (lock1) {
                    System.out.println("Thread 2");
                }
            }
        }).start();
    }
}

线程转储中的死锁表现:

Found one Java-level deadlock:
=============================
Thread 1:
  waiting to lock monitor 0x00007f03fc005000 (object 0x0000000749b623a0, a java.lang.Object),
  which is held by Thread 2

Thread 2:
  waiting to lock monitor 0x00007f03fc005100 (object 0x0000000749b623b0, a java.lang.Object),
  which is held by Thread 1

解决方案:

  • 使用 jstack 检测死锁。
  • 调整锁顺序,避免循环依赖。

3.2 线程池耗尽

示例代码:

ExecutorService executor = Executors.newFixedThreadPool(2);
for (int i = 0; i < 10; i++) {
    executor.submit(() -> {
        try { Thread.sleep(1000); } catch (InterruptedException e) {}
    });
}

线程转储表现:

"pool-1-thread-1" #12 prio=5 os_prio=0 tid=0x00007f03fc005000 nid=0xa9b06 waiting on condition [0x00007f0438dfc000]
   java.lang.Thread.State: WAITING (parking)

解决方案:

  • 增大线程池大小或使用 ThreadPoolExecutor 动态调整。
  • 使用有界队列避免OOM。

3.3 数据库连接泄漏

Druid 配置优化:

spring:
  datasource:
    druid:
      initial-size: 5
      min-idle: 5
      max-active: 20
      max-wait: 60000
      time-between-eviction-runs-millis: 60000
      min-evictable-idle-time-millis: 300000
      validation-query: SELECT 1
      test-while-idle: true

监控SQL泄漏:

// 在代码中显式关闭连接
try (Connection conn = dataSource.getConnection()) {
    // SQL操作
} // 自动关闭

4. 线程转储分析工具

VisualVM(可视化分析线程状态)

FastThread(在线分析工具)

Eclipse MAT(分析线程引用关系)

5. 总结

通过分析线程转储,我们可以:

  • 发现死锁、线程阻塞等问题。
  • 优化线程池和数据库连接池配置。
  • 诊断应用崩溃或性能下降的原因。

最佳实践:

  • 定期采集线程转储(尤其在应用卡顿时)。
  • 结合日志和监控(如Prometheus + Grafana)全面分析。
  • 使用自动化工具(如Arthas)进行动态诊断。

到此这篇关于一文带你深入解析Java应用线程转储的文章就介绍到这了,更多相关Java线程转储内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MyBatis中使用分页插件PageHelper实现分页功能

    MyBatis中使用分页插件PageHelper实现分页功能

    分页是经常使用的功能,本文主要介绍了Mybatis中处理特殊SQL处理逻辑,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-06-06
  • 如何把idea中的项目导入github仓库中(图文详解)

    如何把idea中的项目导入github仓库中(图文详解)

    这篇文章主要介绍了如何把idea中的项目导入github仓库中,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-07-07
  • springcloud 熔断监控Hystrix Dashboard和Turbine

    springcloud 熔断监控Hystrix Dashboard和Turbine

    这篇文章主要介绍了springcloud 熔断监控Hystrix Dashboard和Turbine,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • Spring Security实现动态路由权限控制方式

    Spring Security实现动态路由权限控制方式

    这篇文章主要介绍了Spring Security实现动态路由权限控制方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-08-08
  • idea创建项目没有webapp文件夹的解决方法

    idea创建项目没有webapp文件夹的解决方法

    本文主要介绍了idea创建项目没有webapp文件夹的解决方法,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • Java实现两个日期相减等于天数

    Java实现两个日期相减等于天数

    这篇文章主要介绍了Java两个日期相减等于天数的实现方式,本文通过两种方式结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-09-09
  • SpringBoot静态资源css,js,img配置方案

    SpringBoot静态资源css,js,img配置方案

    这篇文章主要介绍了SpringBoot静态资源css,js,img配置方案,下文给大家分享了三种解决方案,需要的朋友可以参考下
    2017-07-07
  • 东八区springboot如何配置序列化

    东八区springboot如何配置序列化

    本文主要介绍了东八区springboot如何配置序列化,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • Spring boot实现一个简单的ioc(2)

    Spring boot实现一个简单的ioc(2)

    这篇文章主要为大家详细介绍了Spring boot实现一个简单ioc的第二篇,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-04-04
  • 详解Maven settings.xml配置(指定本地仓库、阿里云镜像设置)

    详解Maven settings.xml配置(指定本地仓库、阿里云镜像设置)

    这篇文章主要介绍了详解Maven settings.xml配置(指定本地仓库、阿里云镜像设置),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12

最新评论