juc06-BLOCKED状态
前言
java线程模型中有5种
线程状态。五种状态之间可以进行转换。
强调一下 BLOCKED
状态跟 I/O
的阻塞是不同的,它不是一般意义上的阻塞,而是特指被 synchronized 块阻塞
,即是跟线程同步有关的一个状态。
BLOCKED(阻塞)
简单定义为:
A thread that is blocked waiting for a monitor lock is in this state.
一个正在阻塞等待一个监视器锁的线程处于这一状态。
两种阻塞的情况:
第一种:
A thread in the blocked state is waiting for a monitor lock to enter a synchronized block/method.
一个处于 blocked 状态的线程正在等待一个监视器锁以进入一个同步的块或方法。
第二种:
A thread in the blocked state is waiting for a monitor lock to reenter a synchronized block/method after calling Object.wait。
一个处于 blocked 状态的线程正在等待一个监视器锁,在其调用 Object.wait 方法之后,以再次进入一个同步的块或方法。
进入(enter)同步块时阻塞
监视器锁用于同步访问,以达到多线程间的互斥。所以一旦一个线程获取锁进入同步块,在其出来之前,如果其它线程想进入,就会因为获取不到锁而阻塞在同步块之外,这时的状态就是 BLOCKED。
注:这一状态的进入及解除都不受我们控制,当锁可用时,线程即从阻塞状态中恢复。
wait 之后重进入(reenter)同步块时阻塞
举个场景:
- 调用 wait 方法必须在同步块中,即是要先获取锁并进入同步块,这是第一次 enter。
而调用 wait 之后则会释放该锁,并进入此锁的等待队列(wait set)中。 - 当收到其它线程的 notify 或 notifyAll 通知之后,等待线程并不能立即恢复执行,因为停止的地方是在同步块内,而锁已经释放了,所以它要重新获取锁才能再次进入(reenter)同步块,然后从上次 wait 的地方恢复执行。
这是第二次 enter,所以叫 reenter。
但锁并不会优先给它,该线程还是要与其它线程去竞争锁,这一过程跟 enter 的过程其实是一样的,因此也可能因为锁已经被其它线程据有而导致 BLOCKED。
模拟 BLOCKED 代码
1 |
|
总结
为什么专门提这一块,因为后面在排查问题,查看错误的vm_log时,线常会看到BLOCKED状态,需要了解这个状态代表什么,排查问题才看的问是什么意思。