Java多线程系列 -- 01. Java线程状态及切换

线程状态

操作系统线程状态

操作系统线程只有三种状态:就绪状态、运行状态和阻塞状态。

Java线程状态

Java Thread 有 6 种状态,分别如下:

  • NEW

    至今尚未启动的线程处于这种状态。

  • RUNNABLE

    可运行线程的线程状态。处于可运行状态的某一线程正在 Java 虚拟机中运行,但它可能正在等待操作系统中的其他资源,比如处理器。

  • BLOCKED

    受阻塞并且正在等待监视器锁的某一线程的线程状态。处于受阻塞状态的某一线程正在等待监视器锁,以便进入一个同步的块/方法,或者在调用 Object.wait 之后再次进入同步的块/方法。

  • WAITING

    某一等待线程的线程状态。某一线程因为调用下列方法之一而处于等待状态:

    • 不带超时值的 Object.wait
    • 不带超时值的 Thread.join
    • LockSupport.park

      处于等待状态的线程正等待另一个线程,以执行特定操作。

      例如,已经在某一对象上调用了 Object.wait() 的线程正等待另一个线程,以便在该对象上调用 Object.notify() 或 Object.notifyAll()。已经调用了 Thread.join() 的线程正在等待指定线程终止。

  • TIMED_WAITING

    具有指定等待时间的某一等待线程的线程状态。某一线程因为调用以下带有指定正等待时间的方法之一而处于定时等待状态:

    • Thread.sleep
    • 带有超时值的 Object.wait
    • 带有超时值的 Thread.join
    • LockSupport.parkNanos
    • LockSupport.parkUntil
  • TERMINATED

    已终止线程的线程状态。线程已经结束执行。

在给定时间点上,一个线程只能处于一种状态。这些状态是虚拟机状态,它们并没有反映所有操作系统线程状态。

操作系统线程状态和Java线程状态的对应关系

Java Thread State 操作系统线程
RUNNABLE 就绪状态和运行状态
BLOCKED 阻塞状态
WAITING 阻塞状态
TIMED_WAITING 阻塞状态
NEW 和 TERMINATED 不是线程状态

Java线程状态切换

首先看一下线程状态切换图:(图片来自 skywang12345的博客)

结合上图来讲解下线程状态:

  1. 新建状态(New) : 线程对象被创建后,就进入了新建状态。例如,Thread thread = new Thread()。

  2. 就绪状态(Runnable): 也被称为“可执行状态”。线程对象被创建后,其它线程调用了该对象的start()方法,从而来启动该线程。例如,thread.start()。处于就绪状态的线程,随时可能被CPU调度执行。

  3. 运行状态(Running) : 可运行状态(runnable)的线程获得了cpu 时间片(timeslice),执行程序代码。需要注意的是,线程只能从就绪状态进入到运行状态。

  4. 阻塞状态(Blocked) : 阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪(Runable)状态,才有机会转到运行状态。阻塞的情况分三种:

    • 等待阻塞 – 运行(running)的线程执行o.wait()方法,JVM会把该线程放入等待队列(waitting queue)中,同时wait()会释放持有的锁。
    • 同步阻塞 – 线程在获取synchronized同步锁失败(因为锁被其它线程所占用),它会进入同步阻塞状态。JVM会把该线程放入锁池(lock pool)中。
    • 其他阻塞 – 通过调用线程的sleep()或join()或发出了I/O请求时,线程会进入到阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪(Runable)状态。
  5. 死亡状态(Dead/Terminated) : 线程执行完了或者因异常退出了run()方法,该线程结束生命周期。

参考资料:

skywang12345 Java多线程系列–“基础篇”01之 基本概念

林炳文Evankaka Java多线程学习(吐血超详细总结)

jdk-枚举 Thread.State