文章目录
前言
本文介绍一下在 Java 语言中线程的几种状态。
一、线程状态的六种状态
Java 的线程类中一共枚举了6种线程状态:
如图:
1. NEW(初始状态)
线程被创建出来了,但还没有调用 start() 方法
Thread thread = new Thread();
System.out.println(thread.getState()); // NEW
从上面代码可以看到,只是创建了线程的实例而已,还没有调用 start
方法,所以此时线程是处于 NEW
状态。
这里引申出另一个问题:能否重复调用一个线程的 start
方法呢?
首先我们先看下 start
方法的源码,我们可以看到在 Thread
类内部有一个变量 threadStatus
。线程在调用 start
方法时,会先判断 threadStatus
变量等不等于0,如果 threadStatus
不等于0,则会抛出异常。而在源码注释中, 0 所表示的状态刚好就是 NEW(新建状态)。
所以我们可以得出结论:在程序中是不能重复调用线程的 start
方法,重复调用时,会抛出 IllegalThreadStateException
异常
2. RUNNABLE(运行状态)
若线程调用了 start() ,线程就处于等待运行的状态。
Runnable
包含了操作系统中线程状态的 Running
和 Ready
,也就是说处于此状态下的线程可能正在运行(Running
状态);也可能正在等待 CPU 为它分配时间片(Ready
),或者等待系统资源(如 I/O)。
3. BLOCKED(阻塞状态)
由于某种原因导致正在运行的线程放弃了CPU资源并且暂停了执行,此时线程进入了阻塞状态。
BLOCKED
状态表示该线程已被挂起,正在等待别的线程释放锁,从而才能进入同步代码块或者同步方法之中。在别的线程释放锁之前,该线程无法进入队列争抢CPU资源。
4. WAITING(等待状态)
当线程处于该状态时,需要一直等待直到其他线程做出一些特定动作(通知或中断)。
处于此状态的线程不会被CPU分配时间片,需要等待其他线程执行某个特定的动作去唤醒。
根据源码的注释可以知道以下几个常见的无参方法会造成线程进入该状态:
Object.wait()
Thread.join()
LockSupport.park()
5. TIME_WAITING(超时等待状态)
处于在该状态的线程可以在指定的时间后自行返回,而不是像 WAITING 那样一直等待。
处于此状态的线程也不会被 CPU 分配时间片,不过无需让其他线程来进行唤醒,只需要等待给定的时间就会自动唤醒。
同样,根据源码的注释我们可以知道,以下常见的带参数的方法会让线程进入 TIME_WAITTING
状态:
Thread.sleep(long mills)
Object.wait(long mills)
Thread.join(long mills)
6. TERMINATED(终止状态)
表示该线程已经运行完毕。
二、线程状态变迁图
图源:《Java并发编程艺术》一书
线程之间的几种状态变化可以用下图来总结:
三、总结
本文介绍在 Java 中线程的几种状态,同时给出了线程状态变迁图,我们需要知道的是调用什么样的方法会让线程进入什么样的状态即重点掌握线程状态变迁图的内容。