探秘jstack:解决Java应用线程问题的利器

简介: 探秘jstack:解决Java应用线程问题的利器

基本介绍

jstack 是 JDK 自带的工具,用于生成 Java 进程的线程快照,可以用来诊断 Java 应用程序的性能问题和死锁情况。下面是 jstack 的基本用法和参数介绍:

  1. 命令格式
jstack [option] pid
  1. 参数说明
  • pid:Java 进程的进程号,用于指定要生成线程快照的目标 Java 进程,可以使用 jps 命令查看。
  • option:可选参数,用于指定 jstack 的具体操作选项,常用的选项包括:
  • -l:除堆栈外,显示关于锁的附加信息。
  • -F:当 “jstack pid” 没有响应时强制输出线程栈。
  • -m:如果生成线程快照失败,则尝试使用操作系统级别的线程堆栈。
  • -h--help:显示 jstack 命令的帮助信息。
  1. 将快照信息转存至 dump(一般都是以 dump 结尾)文件:jstack pid> 绝对或者相对路径的 dump 文件目录
  2. 使用示例
  • 生成线程快照:jstack 12345
  • 生成线程快照并显示关于锁的信息:jstack -l 12345
  • 强制生成线程快照:jstack -F 12345
  1. 常见场景
  • 查看 Java 进程中线程的运行状态、堆栈信息,以定位性能瓶颈或死锁问题。
  • 在定位 Java 进程出现假死、性能下降等问题时,生成线程快照进行分析。

实战场景:死锁定位

运行示例程序如下:

package world.xuewei;
/**
 * 死锁示例程序
 *
 * @author 薛伟
 */
public class DeadLockDemo {
    private static final String A = "A";
    private static final String B = "B";
    public static void main(String[] args) {
        new DeadLockDemo().deadLock();
    }
    private void deadLock() {
        new Thread(() -> {
            synchronized (A) {
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (B) {
                    System.out.println("1");
                }
            }
        }, "Thread1").start();
        new Thread(() -> {
            synchronized (B) {
                synchronized (A) {
                    System.out.println("2");
                }
            }
        }, "Thread2").start();
    }
}

实战场景:查看线程状态

运行示例程序如下:

package world.xuewei;
import java.util.concurrent.TimeUnit;
/**
 * 线程的生命周期
 *
 * @author 薛伟
 */
public class ThreadLifeDemo {
    public static final Object LOCK = new Object();
    public static void main(String[] args) {
        // 该线程不断睡眠
        new Thread(() -> {
            while (true) sleepSecond(100);
        }, "T1").start();
        // 该线程在 ThreadLifeDemo.class 等待
        new Thread(() -> {
            while (true) {
                synchronized (ThreadLifeDemo.class) {
                    try {
                        ThreadLifeDemo.class.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }, "T2").start();
        // 该线程锁住 LOCk 后不会释放
        new Thread(() -> {
            synchronized (LOCK) {
                while (true) sleepSecond(100);
            }
        }, "T3").start();
        // 该线程锁住 LOCk 后不会释放
        new Thread(() -> {
            synchronized (LOCK) {
                while (true) sleepSecond(100);
            }
        }, "T4").start();
    }
    /**
     * 睡眠指定秒数
     */
    public static void sleepSecond(int second) {
        try {
            TimeUnit.SECONDS.sleep(second);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}


相关文章
|
1天前
|
Java
Java一分钟之-并发编程:线程间通信(Phaser, CyclicBarrier, Semaphore)
【5月更文挑战第19天】Java并发编程中,Phaser、CyclicBarrier和Semaphore是三种强大的同步工具。Phaser用于阶段性任务协调,支持动态注册;CyclicBarrier允许线程同步执行,适合循环任务;Semaphore控制资源访问线程数,常用于限流和资源池管理。了解其使用场景、常见问题及避免策略,结合代码示例,能有效提升并发程序效率。注意异常处理和资源管理,以防止并发问题。
18 2
|
1天前
|
安全 Java 容器
Java一分钟之-并发编程:线程安全的集合类
【5月更文挑战第19天】Java提供线程安全集合类以解决并发环境中的数据一致性问题。例如,Vector是线程安全但效率低;可以使用Collections.synchronizedXxx将ArrayList或HashMap同步;ConcurrentHashMap是高效线程安全的映射;CopyOnWriteArrayList和CopyOnWriteArraySet适合读多写少场景;LinkedBlockingQueue是生产者-消费者模型中的线程安全队列。注意,过度同步可能影响性能,应尽量减少共享状态并利用并发工具类。
13 2
|
1天前
|
Java 程序员 调度
Java中的多线程编程:基础知识与实践
【5月更文挑战第19天】多线程编程是Java中的一个重要概念,它允许程序员在同一时间执行多个任务。本文将介绍Java多线程的基础知识,包括线程的创建、启动和管理,以及如何通过多线程提高程序的性能和响应性。
|
1天前
|
Java
深入理解Java并发编程:线程池的应用与优化
【5月更文挑战第18天】本文将深入探讨Java并发编程中的重要概念——线程池。我们将了解线程池的基本概念,应用场景,以及如何优化线程池的性能。通过实例分析,我们将看到线程池如何提高系统性能,减少资源消耗,并提高系统的响应速度。
11 5
|
1天前
|
算法 搜索推荐 Java
滚雪球学Java(33):数组算法大揭秘:应用案例实战分享
【5月更文挑战第8天】🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
29 8
滚雪球学Java(33):数组算法大揭秘:应用案例实战分享
|
2天前
|
消息中间件 安全 Java
理解Java中的多线程编程
【5月更文挑战第18天】本文介绍了Java中的多线程编程,包括线程和多线程的基本概念。Java通过继承Thread类或实现Runnable接口来创建线程,此外还支持使用线程池(如ExecutorService和Executors)进行更高效的管理。多线程编程需要注意线程安全、性能优化和线程间通信,以避免数据竞争、死锁等问题,并确保程序高效运行。
|
2天前
|
存储 Java
【Java】实现一个简单的线程池
,如果被消耗完了就说明在规定时间内获取不到任务,直接return结束线程。
10 0
|
2天前
|
安全 Java 容器
深入理解Java并发编程:线程安全与性能优化
【5月更文挑战第18天】随着多核处理器的普及,并发编程变得越来越重要。Java提供了丰富的并发编程工具,如synchronized关键字、显式锁Lock、原子类、并发容器等。本文将深入探讨Java并发编程的核心概念,包括线程安全、死锁、资源竞争等,并分享一些性能优化的技巧。
|
SQL 存储 Java
Java 应用与数据库的关系| 学习笔记
快速学习 Java 应用与数据库的关系。
172 0
Java 应用与数据库的关系| 学习笔记
|
SQL 存储 Java
Java 应用与数据库的关系| 学习笔记
快速学习 Java 应用与数据库的关系。
167 0
Java 应用与数据库的关系| 学习笔记