之前的文章讲过java api层面的线程状态有六种,其中阻塞占3种,BLOCKED , WAITING , TIMED_WAITING 都是 Java API 层面对【阻塞状态】的细分
下面讲讲各个线程状态之间的转换情况
线程状态

假设有线程 Thread t
情况 1 NEW–> RUNNABLE
- 当调用 t.start() 方法时,由 NEW --> RUNNABLE
情况 2 RUNNABLE <–> WAITING
t 线程用 synchronized(obj) 获取了对象锁后
- 调用 obj.wait() 方法时,t线程从 RUNNABLE --> WAITING
- 调用 obj.notify() , obj.notifyAll() , t.interrupt() 时
- 竞争锁成功,t 线程从 WAITING --> RUNNABLE
- 竞争锁失败,t 线程从 WAITING --> BLOCKED
@Slf4j
public class TestWait {
final static Object obj = new Object();
public static void main(String[] args) throws InterruptedException{
new Thread(() -> {
synchronized (obj) {
log.debug("执行....");
try {
obj.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
log.debug("其它代码...."); // 断点
}
},"t1").start();
new Thread(() -> {
synchronized (obj) {
log.debug("执行....");
try {
obj.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
log.debug("其它代码...."); // 断点
}
},"t2").start();
Thread.sleep(5000);
log.debug("唤醒 obj 上其它线程");
synchronized (obj) {
obj.notifyAll(); // 唤醒obj上所有等待线程 断点
}
}
}
- 先运行主线程,此时t1,t2均为等待状态

-
当唤醒线程后,t1 t2进入阻塞状态

-
释放锁后,t2优先获取到锁,t1进入entryList等待唤醒
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e8H93KkV-1655560885097)(D:\Note\面试学习笔记.assets\image-20220618203921043.png)]](/https://i-blog.csdnimg.cn/blog_migrate/504ae1f41eb68cb637b603062daa51ee.png)
-
t2锁释放后,t1就获取到锁,进入运行状态

情况 3 RUNNABLE <–> WAITING
- 当前线程调用 t.join() 方法时,当前线程从 RUNNABLE --> WAITING
- 注意是当前线程在t 线程对象的监视器上等待
- t 线程运行结束,或调用了当前线程的 interrupt() 时,当前线程从 WAITING --> RUNNABLE
- 当前线程调用 LockSupport.park() 方法会让当前线程从 RUNNABLE --> WAITING
- 调用 LockSupport.unpark(目标线程) 或调用了线程 的 interrupt() ,会让目标线程从 WAITING --> RUNNABLE
情况 4 RUNNABLE <–> TIMED_WAITING
t 线程用 synchronized(obj) 获取了对象锁后
-
调用 obj.wait(long n) 方法时,t 线程从 RUNNABLE --> TIMED_WAITING
-
t 线程等待时间超过了 n 毫秒,或调用 obj.notify() , obj.notifyAll() , t.interrupt() 时
-
竞争锁成功,t 线程从 TIMED_WAITING --> RUNNABLE
-
竞争锁失败,t 线程从 TIMED_WAITING --> BLOCKED
-
-
当前线程调用 t.join(long n) 方法时,当前线程从 RUNNABLE --> TIMED_WAITING
- 注意是当前线程在t 线程对象的监视器上等待
-
当前线程等待时间超过了 n 毫秒,或t 线程运行结束,或调用了当前线程的 interrupt() 时,当前线程从TIMED_WAITING --> RUNNABLE
-
当前线程调用 Thread.sleep(long n) ,当前线程从 RUNNABLE --> TIMED_WAITING
-
当前线程等待时间超过了 n 毫秒,当前线程从 TIMED_WAITING --> RUNNABLE
-
当前线程调用 LockSupport.parkNanos(long nanos) 或 LockSupport.parkUntil(long millis) 时,当前线程从 RUNNABLE --> TIMED_WAITING
-
调用 LockSupport.unpark(目标线程) 或调用了线程 的 interrupt() ,或是等待超时,会让目标线程从TIMED_WAITING–>RUNNABLE
情况 5 RUNNABLE <–> BLOCKED
- t 线程用 synchronized(obj) 获取了对象锁时如果竞争失败,从 RUNNABLE --> BLOCKED
- 持 obj 锁线程的同步代码块执行完毕,会唤醒该对象上所有 BLOCKED 的线程重新竞争,如果其中 t 线程竞争成功,从 BLOCKED --> RUNNABLE ,其它失败的线程仍然 BLOCKED
情况6 RUNNABLE <–> TERMINATED
当前线程所有代码运行完毕,进入 TERMINATED
本文详细介绍了Java中线程的六种状态及其转换,包括NEW、RUNNABLE、BLOCKED、WAITING、TIMED_WAITING和TERMINATED。通过实例展示了线程在不同状态间的切换,如调用wait()、notify()、join()、sleep()等方法时的状态变化,帮助理解Java多线程的运行机制。

312

被折叠的 条评论
为什么被折叠?



