博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java 几种线程状态之间的相互关系
阅读量:6445 次
发布时间:2019-06-23

本文共 2037 字,大约阅读时间需要 6 分钟。

Java Thread 可能处在以下几种状态

Java Doc 里通过一个枚举类型 Enum<Thread.State> 来定义。

线程可以处于以下状态之一:

图片描述

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

线程在给定时间点只能处于一种状态。这些状态是虚拟机状态,不反映任何操作系统线程状态。

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

transitive-graph.PNG

由 生成的传递图,显示哪些线程阻塞了哪些线程

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/

转载地址:http://ykvwo.baihongyu.com/

你可能感兴趣的文章
磁盘缓存专题之一:缓存命中和缓存未命中&缓存与缓冲间的差异
查看>>
android应用崩溃的调试方法
查看>>
oracle9i卸载
查看>>
Count and Say
查看>>
第十七章 特殊成员_使用typedef简化函数指针的声明
查看>>
Java NIO使用及原理分析 (四)
查看>>
poj3903
查看>>
EWM RF 开发常用代码
查看>>
shell编程 逻辑分支
查看>>
对结果集进行分页显示
查看>>
构造函数初始化列表
查看>>
关于核实PDF.NET会员用户信息的公告
查看>>
MySql 字符串连接
查看>>
C# 实现导出网站功能
查看>>
Visual C++ 2008入门经典 第十章标准模板库
查看>>
Android SDK 4.2 正式版发布
查看>>
DB4O on Android | Beijing 经验 开放 dB4O 开发 聚会 Party Open | - 好看簿图片博客:用照片记录生活...
查看>>
MySQL Query Cache 小结
查看>>
FPGA中wire与reg类型的区别 .
查看>>
Nginx 反向代理设置
查看>>