面试官:您知道线程的生命周期包括哪几个阶段?
应聘者:
线程的生命周期包含5个阶段,包括:新建
、就绪
、运行
、阻塞
、销毁
。
- 新建: 就是刚使用 new 方法,new 出来的线程;
- 就绪: 就是调用的线程的 start() 方法后,这时候线程处于等待 CPU 分配资源阶段,谁先抢到 CPU 资源,谁开始执行;
- 运行: 当就绪的线程被调度并获得 CPU 资源时,便进入运行状态,run() 方法定义了线程的操作和功能;
- 阻塞: 在运行状态的时候,可能因为某些原因导致运行状态的线程变成了阻塞状态,比如 sleep()、wait() 之后线程就处于了阻塞状态,这个时候需要其他机制将处于阻塞状态的线程唤醒,比如调用 notify 或者 notifyAll() 方法。唤醒的线程不会立刻执行run 方法,它们要再次等待 CPU 分配资源进入运行状态;
- 销毁: 如果线程正常执行完毕后或线程被提前强制性的终止或出现异常导致结束,那么线程就要被销毁,释放资源;
也可以称作 6 个阶段,参考:https://blog.youkuaiyun.com/lzb348110175/article/details/103579044
完整的生命周期图如下:
新建状态
我们来看下面一段代码:
Thread t1 = new Thread();
这里的创建,仅仅是在 JAVA 的这种编程语言层面被创建,而在操作系统层面,真正的线程还没有被创建。只有当我们调用了 start() 方法之后,该线程才会被创建出来,进入 Runnable 状态。只有当我们调用了 start() 方法之后,该线程才会被创建出来。
就绪状态
调用 start() 方法后,JVM 进程会去创建一个新的线程,而此线程不会马上被 CPU 调度运行,进入Running 状态,这里会有一个中间状态,就是 Runnable 状态,你可以理解为等待被 CPU 调度的状态。
t1.start()
用一张图表示如下:
那么处于Runnable状态的线程能发生哪些状态转变?
Runnable 状态的线程无法直接进入 Blocked 状态和 Terminated 状态的。只有处在 Running 状态的线程,换句话说,只有获得 CPU 调度执行权的线程才有资格进入 Blocked 状态和 Terminated 状态,Runnable 状态的线程要么能被转换成 Running 状态,要么被意外终止。
运行状态
当 CPU 调度发生,并从任务队列中选中了某个 Runnable 线程时,该线程会进入Running执行状态,并且开始调用run()方法中逻辑代码。
那么处于Running状态的线程能发生哪些状态转变?
-
被转换成 Terminated 状态,比如调用 stop() 方法;
-
被转换成 Blocked 状态,比如调用了sleep, wait 方法被加入 waitSet 中;
-
被转换成 Blocked 状态,如进行 IO 阻塞操作,如查询数据库进入阻塞状态;
-
被转换成 Blocked 状态,比如获取某个锁的释放,而被加入该锁的阻塞队列中;
-
该线程的时间片用完,CPU 再次调度,进入Runnable 状态;
-
线程主动调用 yield 方法,让出 CPU 资源,进入Runnable 状态。
阻塞状态
Blocked状态的线程能够发生哪些状态改变?
-
被转换成 Terminated 状态,比如调用 stop() 方法,或者是 JVM 意外 Crash;
-
被转换成 Runnable 状态,阻塞时间结束,比如读取到了数据库的数据后;
-
完成了指定时间的休眠,进入到 Runnable 状态;
-
正在 wait 中的线程,被其他线程调用 notify/notifyAll 方法唤醒,进入到 Runnable 状态;
-
线程获取到了想要的锁资源,进入Runnable 状态;
-
线程在阻塞状态下被打断,如其他线程调用了 interrupt 方法,进入到 Runnable 状态;
终止状态
一旦线程进入了Terminated状态,就意味着这个线程生命的终结,哪些情况下,线程会进入到Terminated状态呢?
-
线程正常运行结束,生命周期结束;
-
线程运行过程中出现意外错误;
-
JVM 异常结束,所有的线程生命周期均被结束。
博主写作不易,加个关注呗
求关注、求点赞,加个关注不迷路 ヾ(◍°∇°◍)ノ゙
我不能保证所写的内容都正确,但是可以保证不复制、不粘贴。保证每一句话、每一行代码都是亲手敲过的,错误也请指出,望轻喷 Thanks♪(・ω・)ノ