多线程出bug不知道如何调试?java线程几种常见状态

当你的多线程代码结构很复杂的时候很难找出bug的原因所在,此时我们可以使用getState()方法获取该线程当前的状态,通过观察其状态是阻塞了还是因为没有启动等原因导致的。

状态 描述
NEW 安排了工作,还未开始行动
RUNNABLE 可工作的,又可以分成正在工作中和即将开始工作
BLOCKED 这几个都表示排队等着其他事情
WAITING 这几个都表示排队等着其他事情
TIMED_WAITING 这几个都表示排队等着其他事情
TERMINATED 工作完成了

NEW

此状态说明这个线程已经被创建了,但是没有start()执行任务。

java 复制代码
   Thread t = new Thread(() -> {
       for (int i = 0; i < 1000_0000; i++) {
       }
   }, "状态");
   System.out.println(t.getName() + ": " + t.getState());

RUNNABLE

这个状态表明他已经被创建,但是还没有执行完任务,在这个过程中都是RUNNABLE状态。

java 复制代码
   Thread t = new Thread(() -> {
       for (int i = 0; i < 1000_0000; i++) {
       }
   }, "状态");
   t.start();
   while (t.isAlive()) {
       System.out.println(t.getName() + ": " + t.getState());
   }

TERMINATED

已经执行完任务了,是线程结束的标志

java 复制代码
 Thread t = new Thread(() -> {
     for (int i = 0; i < 1000_0000; i++) {
     }
 }, "状态");
t.start();
t.join();
System.out.println(t.getName() + ": " + t.getState());

WAITING

当他在使用wait,join,sleep方法等待的时候,并且方法没有传入参数,也就是死等的时候,需外部唤醒(如 notify()),处于此状态。

java 复制代码
  Thread t1 = Thread.currentThread();
  Thread t = new Thread(() -> {
      try {
          t1.join();
      } catch (InterruptedException e) {
          throw new RuntimeException(e);
      }
  }, "状态");
  t.start();
  Thread.sleep(10);
  System.out.println(t.getState());

TIMED_WAITING

当他在使用wait,join,sleep方法等待的时候,并且传入参数会进入此状态,超时后自动唤醒或外部提前唤醒,和WAITING的区别就是是否传入参数的区别。

java 复制代码
  Thread t1 = Thread.currentThread();
  Thread t = new Thread(() -> {
      try {
          Thread.sleep(1000);
      } catch (InterruptedException e) {
          throw new RuntimeException(e);
      }
  }, "状态");
  t.start();
  Thread.sleep(10);
  System.out.println(t.getState());

BLOCKED

因为锁竞争导致的等待时,会进入此状态,如果一直是此状态可能是陷入了死锁。

java 复制代码
//这个案例是个死锁。
  Object locker1 = new Object();
  Object locker2 = new Object();
  Thread t1 = new Thread(()->{
      synchronized (locker1) {
          try {
              Thread.sleep(1);
          } catch (InterruptedException e) {
              throw new RuntimeException(e);
          }
          synchronized (locker2) {
          }
      }
      System.out.println("t1执行完毕");
  });

  Thread t2 = new Thread(()->{
      synchronized (locker2) {
          synchronized (locker1) {
          }
      }
      System.out.println("t2执行完毕");
  });
  t1.start();
  t2.start();
  Thread.sleep(100);
  System.out.println(t1.getState());
相关推荐
碎梦归途2 分钟前
23种设计模式-结构型模式之享元模式(Java版本)
java·开发语言·jvm·设计模式·享元模式
神奇侠202420 分钟前
基于PaddleOCR对图片中的excel进行识别并转换成word(一)
python·word·excel·paddleocr
lozhyf21 分钟前
Eureka搭建
java·spring cloud
bigear_码农26 分钟前
python异步协程async调用过程图解
开发语言·python·线程·进程·协程
幽络源小助理31 分钟前
SpringBoot民宿管理系统开发实现
java·spring boot·springboot·民宿系统
东阳马生架构33 分钟前
Nacos简介—1.Nacos使用简介
java
爱发飙的蜗牛38 分钟前
springboot--web开发请求参数接收注解
java·spring boot·后端
码熔burning1 小时前
【MQ篇】RabbitMQ之工作队列模式!
java·分布式·rabbitmq·mq
优雅的落幕1 小时前
Spring--统一数据返回格式与统一异常处理
java·spring·状态模式
BillKu1 小时前
Spring Boot + MyBatis 动态字段更新方法
java·spring boot·mybatis