乐趣区

Java 几种线程状态之间的相互关系

Java Thread 可能处在以下几种状态
Java Doc 里通过一个枚举类型 Enum<Thread.State> 来定义。
线程可以处于以下状态之一:

NEW 尚未启动的线程处于此状态。
RUNNABLE 在 Java 虚拟机中执行的线程处于此状态。
BLOCKED 被阻塞等待监视器锁定的线程处于此状态。
WAITING 无限期等待另一个线程执行特定操作的线程处于此状态。
TIMED_WAITING 正在等待另一个线程执行最多指定等待时间的操作的线程处于此状态。
TERMINATED 已退出的线程处于此状态。

线程在给定时间点只能处于一种状态。这些状态是虚拟机状态,不反映任何操作系统线程状态。
通过示例而不是 Java doc 中给出的正式定义,可以很容易地理解任何令人困惑的概念。如果它们是现实生活中的例子,它可能更具有相关性。我想分享一些可能有助于理解这些线程状态的现实例子。

由 http://fastthread.io 生成的传递图,显示哪些线程阻塞了哪些线程
BLOCKED 阻塞
Java doc 正式将 BLOCKED 状态定义为:“阻塞等待监视器锁的线程处于此状态。”
现实生活中的例子:今天你要去面试。这是您梦寐以求的工作,这是您过去几年一直瞄准的目标。你早上醒来,准备好了,穿上你最好的衣服,在镜子前面看起来很敏锐。现在你走出你的车库,意识到你的妻子已经开车了。在这种情况下,你只有一辆车,那么会发生什么?在现实生活中,可能会发生斗争:-)。在这里你被阻止,因为你的妻子已经开车了。你将无法参加面试。
这是 BLOCKED 状态。用技术术语解释它,你是线程 T1,你的妻子是线程 T2,锁是汽车。T1 在锁(即汽车)上被阻挡,因为 T2 已经获得了这个锁。
提示:线程在等待监视器锁进入同步块 / 方法或在调用 Object#wait()方法后重新输入同步块 / 方法时,将进入 BLOCKED 状态。
WAITING 等候
Java doc 正式将 WAITING 状态定义为:“无限期等待另一个线程执行特定操作的线程处于此状态。”
现实生活中的例子:让我们说几分钟后你的妻子带着车回家了。现在你意识到面试的时间已经到了,而且到达那里还有很长的路要走。因此,您将所有动力都放在汽车的油门踏板上。当允许的速度限制仅为 60 英里 / 小时时,您以 100 英里 / 小时的速度行驶。你不走运,交通警察看到你超过限速,他把你拉到路边。现在你进入 WAITING 状态,我的朋友。你停止开车,坐在车里闲逛,直到警察调查你,然后让你走。基本上,在他让你离开之前,你会陷入等待状态。
用技术术语解释它,你是线程 T1,警察是线程 T2。你释放锁(即你停止开车),然后进入等待状态。直到警察(即 T2)让你离开,你将陷入这种等待状态。
提示:线程在调用以下方法之一时将进入 WAITING 状态:

Object#wait() 未指定超时时间
Thread#join() 未指定超时时间
LockSupport#park()

在对象上调用 Object.wait()的线程处于 WAITING 状态,直到另一个线程调用该对象上的 Object.notify()或 Object.notifyAll()。调用 Thread.join()的线程处于 WAITING 状态,以使指定的线程终止。
TIMED_WAITING 指定时间的等待
Java doc 正式将 TIMED_WAITING 状态定义为:“等待另一个线程在指定的等待时间内执行操作的线程处于此状态。”
现实生活中的例子:尽管所有的戏剧性,你在采访中表现得非常好,给每个人留下了深刻的印象并得到了这份高薪工作。(恭喜!)你回到家里,告诉你的邻居关于这份新工作,以及你对此感到非常兴奋。你的朋友说他也在同一栋办公楼工作。他建议你们两个应该一起开车。你认为这是一个好主意。所以在工作的第一天,你去他家。你把车停在他家门口。你等了 10 分钟,但你的邻居仍然没有出来。你继续开始上班,因为你不想在第一天被推迟。现在这是 TIMED_WAITING。
用技术术语解释它,你是线程 T1,你的邻居是线程 T2。你释放锁定(即停止驾驶汽车)并等待长达 10 分钟。如果你的邻居 T2 在 10 分钟内没出来,你就开始再次开车了。
提示:线程在调用以下方法之一时将进入 TIMED_WAITING 状态:

Thread#sleep()
Object#wait() 指定超时时间
Thread#join() 指定超时时间
LockSupport#parkNanos()
LockSupport#parkUntil()

结论
当有人在分析线程转储时,理解这些不同的线程状态是至关重要的。处于 RUNNABLE,BLOCKED,WAITING 和 TIMED_WATING 状态的线程数是多少?哪些线程被阻止?谁阻止了他们?用于锁定的对象是什么?这些是在线程转储中要分析的一些重要指标。这些详细的线程转储分析可以通过在线工具轻松完成,例如:http://fastthread.io/

退出移动版