• java线程转储分析


    一.线程状态

    java线程状态有两种说法。

    5种状态

    java线程状态:1、新建状态New;2、就绪状态Runnable;3、运行状态Running;4、阻塞状态Blocked;5、死亡状态Dead。

    以上状态是一种概念性说法,参考https://www.runoob.com/note/34745

    6种状态

    Thread.State是一个内部枚举类,定义了6个枚举常量,分别代表Java线程的6种状态。

    这6种状态是在线程转储日志中会见到的。

     注意,runnable包含了就绪和运行中;其他状态只能跟runnable进行转换,只有从runable才能走向terminated;

    二.获取线程转储日志

    首先jps,查询有哪些java进程

     然后用jstack,把此刻的线程日志保存到文件。

    也可以直接打印到控制台

    三.日志格式 

    1. "Thread-1" #13 prio=5 os_prio=0 tid=0x000002b9b6f6d800 nid=0x52e8 waiting for monitor entry [0x000000f8ff9fe000]
    2. java.lang.Thread.State: BLOCKED (on object monitor)
    3. at com.example.mybatisgenerate.thread.ThreadSyc$Task.run(ThreadSyc.java:24)
    4. - waiting to lock <0x000000076b66b4f0> (a [B)
    5. at java.lang.Thread.run(Thread.java:750)
    • "Thread-1":线程名称,建议代码新建线程、或者线程池线程工厂,新建线程都设置自定义名字,方便排查问题。
    • prio:是priority优先级的缩写,表名了当前线程的优先级,取值范围为[1-10],默认为 5。在虚拟机进行线程调度的时候会参考该优先级为线程分配计算资源,这个数值越低越有优先获取到计算资源,一般不设置直接使用默认的优先级
    • os_prio为线程对应系统的优先级。
    • tid:JVM给线程分配的id号
    • nid:本地线程编号NativeID的缩写,对应JVM 虚拟机中线程映射在操作系统中的线程编号。我们可以使用 top 查看进程对应的线程情况进行相关映射。
    • java.lang.Thread.State: BLOCKED 线程状态
    • 再往下是代码执行栈,执行到哪个方法

    四.状态示例

     TIMED_WAITING (sleeping)

    1. public class ThreadSleep {
    2. public static void main(String[] args) throws InterruptedException {
    3. new Thread(() -> {
    4. try {
    5. Thread.sleep(10000);
    6. } catch (InterruptedException e) {
    7. throw new RuntimeException(e);
    8. }
    9. }, "自定义线程").start();
    10. Thread.sleep(20000);
    11. }
    12. }

    sleep方法让线程进入timed_waiting状态,日志括号里备注了是sleeping。 

    线程池的线程回收

    先说结论:当线程数大于线程池核心数,并且线程空闲,此时线程状态是 TIMED_WAITING (parking),超过线程存活时间,线程被回收。

    代码说明:线程池核心数1,最大数2,队列大小1,空闲线程存活时间5s。连续执行三个任务,任务1在执行,任务2在队列里,队列满了---》增加线程数,任务3被执行。(这里面,因为任务睡眠时间长,任务2在队列里肯定是最后被执行的)

    1. public class ThreadSleepPool {
    2. public static ExecutorService executorService = new ThreadPoolExecutor(1, 2, 5000L, TimeUnit.MILLISECONDS,
    3. new LinkedBlockingQueue(1),
    4. new ThreadFactory() {
    5. AtomicInteger atomicInteger = new AtomicInteger();
    6. @Override
    7. public Thread newThread(Runnable r) {
    8. Thread thread = new Thread(r, "自定义线程" + atomicInteger.getAndIncrement());
    9. return thread;
    10. }
    11. });
    12. public static void main(String[] args) throws InterruptedException {
    13. executorService.execute(new Task("1"));
    14. executorService.execute(new Task("2"));
    15. executorService.execute(new Task("3"));
    16. Thread.sleep(20000);
    17. }
    18. public static class Task implements Runnable {
    19. String name;
    20. public Task(String name) {
    21. this.name = name;
    22. }
    23. @Override
    24. public void run() {
    25. try {
    26. System.out.println(name+"执行");
    27. Thread.sleep(10000);
    28. } catch (InterruptedException e) {
    29. throw new RuntimeException(e);
    30. }
    31. }
    32. }
    33. }

     刚开始,两个线程都在sleep,说明任务在执行。

    过一会,线程处于空闲状态,空闲线程getTask得不到任务,就被挂起,parkNanos,变成TIMED_WAITING状态 

    再过一会,空闲线程消失了,已经被回收了。 

     同步锁BLOCKED (on object monitor)

    1. public class ThreadSync {
    2. private static byte[] lock = new byte[1];
    3. public static void main(String[] args) {
    4. final Thread task1 = new Thread(new Task());
    5. final Thread task2 = new Thread(new Task());
    6. task1.start();
    7. task2.start();
    8. }
    9. private static class Task implements Runnable {
    10. @Override
    11. public void run() {
    12. synchronized (lock) {
    13. int i = 0;
    14. while (true) {
    15. i++;
    16. }
    17. }
    18. }
    19. }
    20. }

    线程0拿到同步锁在执行中,状态runnable。线程1堵塞状态,在 monitor Object ,监控对象

     同步锁死锁

    1. public class ThreadDeadSync {
    2. public static byte[] lock1 = new byte[1];
    3. public static byte[] lock2 = new byte[1];
    4. public static void main(String[] args) {
    5. new Thread(() -> {
    6. synchronized (lock1) {
    7. try {
    8. Thread.sleep(3000);
    9. } catch (InterruptedException e) {
    10. throw new RuntimeException(e);
    11. }
    12. synchronized (lock2) {
    13. System.out.println("执行不到");
    14. }
    15. }
    16. }, "自定义线程1").start();
    17. new Thread(() -> {
    18. synchronized (lock2) {
    19. try {
    20. Thread.sleep(3000);
    21. } catch (InterruptedException e) {
    22. throw new RuntimeException(e);
    23. }
    24. synchronized (lock1) {
    25. System.out.println("执行不到");
    26. }
    27. }
    28. }, "自定义线程2").start();
    29. }
    30. }

    两个线程都是Blocked 

    在最下方有死锁分析 。方法执行显示,线程2 locked d0,然后等待另一个锁 waiting to lock b8。线程1locked b8,waiting to lock d0。

     

     ReentrantLock WAITING (parking)

    1. public class ThreadLock {
    2. private static ReentrantLock lock = new ReentrantLock();
    3. public static void main(String[] args) {
    4. final Thread task1 = new Thread(new Task());
    5. final Thread task2 = new Thread(new Task());
    6. task1.start();
    7. task2.start();
    8. }
    9. private static class Task implements Runnable {
    10. @Override
    11. public void run() {
    12. lock.lock();
    13. int i = 0;
    14. while (true) {
    15. i++;
    16. }
    17. }
    18. }
    19. }

     等待锁的线程状态是 WAITING (parking),能看到lock方法最后执行了 LockSupport.park()

     ReentrantLock死锁

    1. public class ThreadDeadLock {
    2. private static ReentrantLock lock1 = new ReentrantLock();
    3. private static ReentrantLock lock2 = new ReentrantLock();
    4. public static void main(String[] args) {
    5. new Thread(() -> {
    6. lock1.lock();
    7. try {
    8. Thread.sleep(3000);
    9. } catch (InterruptedException e) {
    10. throw new RuntimeException(e);
    11. }
    12. lock2.lock();
    13. }, "自定义线程1").start();
    14. new Thread(() -> {
    15. lock2.lock();
    16. try {
    17. Thread.sleep(3000);
    18. } catch (InterruptedException e) {
    19. throw new RuntimeException(e);
    20. }
    21. lock1.lock();
    22. }, "自定义线程2").start();
    23. }
    24. }

    两个线程状态都是  WAITING (parking)

     最下方有死锁分析。这里没有像同步锁显示线程锁住了哪个对象,但是显示了在等待哪个。parking to wait for xxx 

    常用命令

    查询目标线程上下5行信息

    jstack pid |ps -5 "线程名"

     nid16进制转换

    nid=0x38a ,16进制,操作系统的线程编号。

    用C语言的printf函数打印成十进制

    或者用计算器

      

    top查询cpu,十进制nid对应pid

    top -Hp 进程号,883是jps查处的一个java进程,查询进程下的各线程cpu占有率。

    835 836转为16进制就是nid。

    参考

    jstack 命令解读_ghimi的博客-CSDN博客_jstack

  • 相关阅读:
    07-SDRAM :FIFO控制模块
    Kotlin委托属性(1)
    限制命令长度如何反弹shell
    2023最新SSM计算机毕业设计选题大全(附源码+LW)之java高校教师工作量的核算的设计与实现g6ipj
    win10下yolov7 tensorrt模型部署
    windows 安装 zookeeper (图文解说详细版+)
    [Web安全 网络安全]-学习视频分享汇总(持续更新中)
    2022牛客蔚来杯第四场 N K D H A
    Linux编辑器-vim使用
    vue模板语法(下)
  • 原文地址:https://blog.csdn.net/u014203449/article/details/126585767