线程的生命周期
New
当我们使用关键字new 创建一个线程对象时,此时它并不是处于执行状态,因为并没有调用start()启动该线程,这时只是Thread对象的的状态(New), 也即没有start之前, 该线程根本不存在,与我们用new 创建一个普通对象没有什么区别。
New 状态通过start方法进入Runnable状态
Runnable
调用了线程对象的start方法,线程状态此时进入Runnable状态,此时才是真正的在JVM中创建了一个线程, 这是线程不一定立即得到执行,这是由于线程的运行与否和进程一样都要听命于CPU的调度,这个中间状态就是线程的可执行状态(Runnable), 这是也就是说线程具备了执行的资格,但是并没有执行,而是在等待CPU的调度。
注意: 此时的Runnable状态,不会直接进入Blocked或者Terminated状态,即使是在线程执行逻辑中调用wait, sleep或者其他block的IO操作等,也必须先获得CPU的调度执行权才可以, 严格来讲,Runnable的线程只能意外终止或者进入Running状态。
Running
一旦CPU通过轮询或者其他方式从任务可执行队列中选中了线程,此时才可能真正执行自己的逻辑代码(run方法中的代码), 这里需要说明的是一个正在Running的线程事实上也是Runnable的, 但是反过来则不成立。
该状态下线程可发生如下转换:
- 直接进入Terminated状态,比如调用JDK已经不推荐使用的stop方法或者判断某个逻辑标识
- 进入Blocked状态,比如调用sleep或者wait 方法而进入waitSet中
- 进行某个阻塞的IO操作,比如因网络数据的读写而进入Blocked状态
- 获取某个锁资源,从而加入到该锁的阻塞队列中而进入Blocked状态
- 由于CPU的调度器轮询是该线程放弃执行,进入Runnable状态
- 线程主动调用yield方法,放弃CPU的执行权,进入Runnable状态
Blocked
- 调用sleep或者wait 方法而进入waitSet中
- 进行某个阻塞的IO操作,比如因网络数据的读写
- 获取某个锁资源,从而加入到该锁的阻塞队列中
线程在该状态下可以切换的状态如下:
- 直接进入Terminated状态,比如调用JDK已经不推荐使用的stop方法或者意外死亡(JVM Crash)
- 线程阻塞的操作结束, 比如读取到了想要的数据进入到Runnable状态
- 线程完成了指定时间的休眠,进入到Runnable状态
- Wait中的线程被其他线程notify或者notifyAll唤醒, 进入Runnable状态
- 线程获取到了某个锁资源,进入Runnable状态
- 线程在阻塞过程中被打断, 比如其他线程调用了interrupt方法, 进入Runnable状态
Terminated
Terminated 状态是线程的最终状态,在该状态中线程将不会切换到其他的任何状态, 线程进入Terminated状态,意味着该线程的整个生命周期都结束了。
下列情况将会使线程进入Terminated状态:
- 线程运行正常结束,结束生命周期
- 线程运行出错意外结束
- JVM Crash, 导致所有的线程都结束
参考
Java 高并发编程详解 多线程与架构设计 汪文君 著