Java中的线程包括NEW、RUNNABLE、RUNNING、BLOCKED和TERMINATED五种阶段,它们的转换关系可以用下图来表示:
在使用new关键字创建了一个Thread对象时,在调用start方法启动该线程前,线程的状态都为NEW状态,和普通的Java对象并没有什么区别。
线程通过调用start方法从NEW状态进入RUNNABLE状态,此时JVM中真正创建了一个新的线程,线程的运行与否要听令于CPU的调度,这个中间状态称为可执行状态,也就是说它具有执行的资格但是并没有真正的执行。此时的线程只能意外终止或者进入RUNNING状态。
当CPU通过轮询或者其他方式从任务可执行队列中选中了线程,那么此时它才能真正的执行自己的逻辑代码,即从RUNNABLE状态进入RUNNING状态,一个RUNNING状态的线程事实上也是RUNNABLE的,但是反之不成立。在该状态下,线程的状态可以发生以下转换:
- 直接进入TERMINATED状态,比如调用JDK已经不推荐使用的stop方法,或者判断某个逻辑标志
- 进入BLOCKED状态,比如调用了sleep或者wait方法而加入了waitSet中,进入BLOCKED状态(sleep方法不会释放锁的所有权)
- 进行某个阻塞的IO操作,比如因网络数据的读写而进入了BLOCKED状态
- 获取某个锁资源,从而加入该锁的阻塞队列而进入了BLOCKED状态
- 由于CPU的调度器轮询使该线程放弃执行,进入RUNNABLE状态
- 线程主动调用yield方法,放弃CPU执行权,进入RUNNABLE状态
通过sleep、wait方法、阻塞IO操作、获取锁加入该锁的阻塞队列可以使线程从RUNNABLE状态进入BLOCKED状态,这些不再赘述,下面介绍下BLOCKED方法向其它状态的转换:
- 直接进入TERMINATED状态,比如调用JDK已经不推荐使用的stop方法或者意外死亡(JVM Crash)
- 线程阻塞的操作结束,比如读取了想要的数据字节而进入到RUNNABLE状态
- 线程完成了指定时间的休眠,进入到了RUNNABLE状态
- wait中的线程被其它线程notify/notifyAll唤醒,进入RUNNABLE状态
- 线程读取到了某个锁资源,进入到RUNNABLE状态
- 线程在阻塞过程中被打断,比如其它线程调用了interrupt方法,进入RUNNABLE状态
线程在RUNNABLE、RUNNING、BLOCKED状态在发生意外终止或者在RUNNING、RUNNABLE状态时线程调用了stop方法会进入到TERMINATED状态,以下几种情况会使线程进入TERMINATED状态:
- 线程运行正常结束,结束生命周期
- 线程运行出错意外结束
- JVM Crash,导致所有的线程都结束