线程的生命周期
New 新建状态
- 使用关键字new创建一个Thread对象时,此时不处于运行状态,为调用Thread的start方法,线程根本不存在,与关键字new创建普通Java对象相同。
- New状态通过start方法进入Runnable状态。
Runnable 就绪状态
- 线程对象进入Runnable状态必须调用start方法,调用start则创建了一个线程。此时线程并不一定执行,线程的执行取决于CPU状态,如果线程未执行,则该状态就是可执行状态(具备执行的资格),该状态并未执行,而是在等着CPU调度。
- 注意:由于存在Running状态,所以不会直接进入Blocked状态和Dead状态,即使在线程的执行逻辑中调用wait、sleep或者其他block的IO操作等。也必须获得CPU的调度执行权。
- Runnable状态只能意外终止或者进入Running状态
Running 运行状态
- CPU通过轮询或者其他方式从任务可执行队列中选中了线程,那么线程才执行逻辑代码,注:处于Running状态的线程也是Runnable状态,反之则不成立
- 该状态的状态装换:
- 进入Terminated/dead状态,调用JDK已经不推荐使用的Stop方法或者判断某个逻辑标识
- 进入Blocked状态,调用sleep,或者wait方法儿加入了waitSet
- 进入某个阻塞的IO操作,因网络数据的读写而进入Blocked状态
- 获取某个锁资源,从而加入到该锁的阻塞队列中,进入Blocked状态
- 由于CPU的调度器轮询使该线程放弃执行,进入Runnable状态
- 线程主动调用yield方法,放弃CPU执行权,进入Runnable状态
Blocked 阻塞状态
- 线程在Blocked状态中可以切换至如下几个状态:
- 直接进入Terminated状态,比如调用JDK已经不推荐使用的Stop方法或者意外死亡(JVM crash)
- 线程阻塞的操作结束,读取到想要的数据字节,进入到Runnable状态
- 线程完成了指定时间的休眠,进入Runnable状态
- Wait中的线程被其他线程notify/notifyall唤醒,进入Runnable状态
- 线程获取到了某个锁资源,进入Runnable状态
- 线程在阻塞过程中被打断,比如其他线程调用了interrupt方法,进入Runnable状态
Terminated/Dead死亡状态
- 线程的最终状态,该状态不会切换到其他任何状态,线程进入该状态意味着该线程的整个生命周期都结束了
- 使线程进入该状态的情况:
- 线程运行正常结束生命周期
- 线程出错意外结束生命周期
- JVM Crash,导致所有线程结束生命周期