Java线程的五种状态是什么

1. 线程的5种状态

从操作系统层面上,任何线程一般都具有五种状态,即创建、就绪、运行、阻塞、终止。

(1) 新建状态(NEW)

在程序中用构造方法创建一个新线程时,如new Thread(),该线程就是创建状态,此时它已经有了相应的内存空间和其它资源,但是还没有开始执行。

(2) 就绪状态(READ)

启动线程可以通过调用新建线程对象的start()方法实现。当线程启动时,线程就进入就绪状态(runnable)

线程已具备运行条件,但因尚未分配CPU,故进入线程队列排队等待CPU服务。一旦系统选择一个待执行的线程对象,该对象就从等待状态转换为执行状态。系统挑选的动作称之为“CPU调度”。当线程获取了CPU后,便进入运行状态并自动执行其自身的run方法。

(3) 运行状态(RUNNING)

线程进入运行状态是指当就绪状态的线程被调用并占用处理器资源时。此时,自动调用该线程对象的run()方法。

(4) 阻塞状态( BLOCKED)

一个正在执行的线程在某些特殊情况下,如被人为挂起或需要执行耗时的输入输出操作时,将让出CPU并暂时中止自己的执行,进入堵塞状态。

如果使用sleep()、suspend()、wait()等方法在可执行状态下,线程将发生阻塞。阻塞时,线程不能进入排队队列,只能当引起阻塞的原因被消除后,线程转入就绪状态,重新到就绪队列中排队等待,这时被CPU调度选中后会从原来停止的位置开始继续执行。

记住:阻塞被消除后是回到就绪状态,不是运行状态。

(5) 死亡状态(TERMINATED)

当线程调用stop()、destory()或run()并执行完毕后,该线程将处于终结状态。处于死亡状态的线程不具有继续运行的能力。

2. Java线程的6种状态

Java中线程的生命周期分为6种状态。Thread类有一个实例属性和一个实例方法专门用于保存和获取线程的状态。其中,用于保存线程Thread实例状态的实例属性为:

// 以整数的形式保存线程的状态
private volatile int threadStatus = 0;
// 返回当前线程的状态,一个枚举类型值
public State getState() {
    return sun.misc.VM.toThreadState(threadStatus);
}

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

public enum State {
    // 新建状态
    NEW,
    // 运行状态
    RUNNABLE,
    /**
     * 阻塞状态
     * Object.wait
     */
    BLOCKED,
    /**
     *  等待状态
     *  Object.wait
     *  Thread.join
     *  LockSupport.park
     */
    WAITING,
    /**
     *  限时等待状态
     *  Thread.sleep
     *  Object.wait
     *  Thread.join
     *  LockSupport.parkUntil
     *  LockSupport.parkNanos
     */
    TIMED_WAITING,
    // 终止状态
    TERMINATED;
}

有4种是比较常见的状态,它们是:NEW(新建)状态、RUNNABLE(可执行)状态、TERMINATED(终止)状态、TIMED_WAITING(限时等待)状态。

(1) NEW状态

Java源码对NEW状态的说明是:创建成功但是没有调用start()方法启动的Thread线程实例都处于NEW状态。

当然,并不是Thread线程实例的start()方法一经调用,其状态就从NEW状态到RUNNABLE状态,此时并不意味着线程立即获取CPU时间片并且立即执行,中间需要一系列操作系统的内部操作。

(2) RUNNABLE状态

当调用了Thread实例的start()方法后,下一步如果线程获取CPU时间片开始执行,JVM将异步调用线程的run()方法执行其业务代码。那么在run()方法被异步调用之前,JVM做了哪些事情呢?当Java线程的Thread实例的start()方法被调用后,操作系统中的对应线程进入的并不是运行状态,而是就绪状态,而Java线程并没有这个就绪状态。操作系统中线程的就绪状态是什么状态的呢?JVM的线程状态与其幕后的操作系统线程状态之间的转换关系简化后如图:

一个操作系统线程如果处于就绪状态,就表示“万事俱备,只欠东风”,即该线程已经满足执行条件,但是还不能执行。处于就绪状态的线程需要等待系统的调度,一旦就绪状态被系统选中,获得CPU时间片,线程就开始占用CPU,开始执行线程的代码,这时线程的操作系统状态发生了改变,进入了运行状态。

当操作系统中运行状态的线程用完CPU时间片后,它会返回到就绪状态,等待下一次CPU调度。就这样,操作系统线程在就绪状态和执行状态之间被系统反复地调度,这种情况会一直持续,直到线程的代码逻辑执行完成或者异常终止。这时线程的操作系统状态又发生了改变,进入线程的最后状态——TERMINATED状态。

就绪状态和运行状态都是操作系统中的线程状态。在Java语言中,并没有细分这两种状态,而是将这两种状态合并成同一种状态——RUNNABLE状态。因此,在Thread.State枚举类中,没有定义线程的就绪状态和运行状态,只是定义了RUNNABLE状态。这就是Java线程状态和操作系统中线程状态不同的地方。

当Thread实例处于NEW状态时,一旦调用start()方法,该线程将转换为RUNNABLE状态。尽管如此,线程的run()方法不一定会马上被并发执行,需要在线程获取了CPU时间片之后才真正启动并发执行。

(3) TERMINATED状态

当处于RUNNABLE状态的线程执行完run()方法后,它会进入终止状态TERMINATED。当然,如果在run()方法执行过程中发生了运行时异常而没有被捕获,run()方法将被异常终止,线程也会变成TERMINATED状态。

(4) TIMED_WAITING状态

线程处于一种特殊的等待状态,准确地说,线程处于限时等待状态。能让线程处于限时等待状态的操作大致有以下几种:

  • Thread.sleep(int n):使得当前线程进入限时等待状态,等待时间为n毫秒。

  • Object.wait():带时限的抢占对象的monitor锁。

  • Thread.join():带时限的线程合并。

  • LockSupport.parkNanos():让线程等待,时间以纳秒为单位。

  • LockSupport.parkUntil():让线程等待,时间可以灵活设置。

3. Java线程状态的转换

以上就是Java线程的五种状态是什么的详细内容,更多请关注其它相关文章!