区块链技术博客
www.b2bchain.cn

多个Java线程似乎锁定了同一监视器? – java程序员分享

本文介绍了多个Java线程似乎锁定了同一监视器? – java程序员分享,有助于帮助完成毕业设计以及求职,是一篇很好的资料。

对技术面试,学习经验等有一些体会,在此分享。

在Java threaddump中,我发现了以下内容:

"TP-Processor184" daemon prio=10 tid=0x00007f2a7c056800 nid=0x47e7 waiting for monitor entry [0x00007f2a21278000]    java.lang.Thread.State: BLOCKED (on object monitor)     at org.apache.jackrabbit.core.state.SharedItemStateManager.getNonVirtualItemState(SharedItemStateManager.java:1725)     - locked <0x0000000682f99d98> (a org.apache.jackrabbit.core.state.SharedItemStateManager)     at org.apache.jackrabbit.core.state.SharedItemStateManager.getItemState(SharedItemStateManager.java:257)  "TP-Processor137" daemon prio=10 tid=0x00007f2a7c00f800 nid=0x4131 waiting for monitor entry [0x00007f2a1ace7000]    java.lang.Thread.State: BLOCKED (on object monitor)     at org.apache.jackrabbit.core.state.SharedItemStateManager.getNonVirtualItemState(SharedItemStateManager.java:1725)     - locked <0x0000000682f99d98> (a org.apache.jackrabbit.core.state.SharedItemStateManager)     at org.apache.jackrabbit.core.state.SharedItemStateManager.getItemState(SharedItemStateManager.java:257) 

这里的要点是两个线程都已锁定监视器 (不管它们现在正在等待其他两个监视器)。

在查看“线程转储分析器”时,选中该监视器后,实际上会在底部显示“线程锁定监视器:2”,并显示“2线程锁定”。截图请参见https://lh4.googleusercontent.com/-fCmlnohVqE0/T1D5lcPerZI/AAAAAAAAD2c/vAHcDiGOoMo/s971/locked_by_two_threads_3.png,我不允许在此处粘贴图像。

这是否意味着对于监视锁定信息而言,线程转储不是原子的?我无法想象这真的是JVM(1.6.0_26-b03)的锁定错误。

Can several threads hold a lock on the same monitor in Java?中已经提出了类似的问题,但是对我的回答并未看到多个线程锁定同一监视器的真正意义,即使它们可能正在等待其他线程。

2014年5月13日更新:

较新的问题Multiple threads hold the same lock?具有可重现此行为的代码,@ rsxg已在此处按照其答案提交了相应的错误报告https://bugs.openjdk.java.net/browse/JDK-8036823。

参考方案

分析严重竞争的锁时,您可能会在JVM的堆栈跟踪例程中遇到表面上的错误-它可能与this bug相同或不同。

事实是,您的两个线程都没有实际获得对SharedItemStateManager的锁定,正如您从它们正在报告waiting for monitor entry的事实中所看到的那样。错误是,在这两种情况下,堆栈跟踪都应该向上报告,而不是waiting to lock,它们应该报告locked

分析此类奇怪堆栈跟踪时的解决方法是,始终检查声称拥有locked一个对象的线程是否也没有等待获取同一对象的锁。

(不幸的是,此分析需要将堆栈跟踪中的行号与源代码交叉引用,因为waiting for monitor entry标头中的数字与堆栈跟踪中的locked行之间没有关系。根据this Oracle document, 0x00007f2a21278000行是线程的有效堆栈区域的估计值,因此它看起来像是监视器ID,但不是-您可以看到您给的两个线程位于堆栈中的不同地址)。

试图绕过Java并发问题,并且很难理解线程池,线程以及它们正在执行的可运行“任务”之间的关系。如果我创建一个有10个线程的线程池,那么我是否必须将相同的任务传递给池中的每个线程,或者池化的线程实际上只是与任务无关的“工人无人机”可用于执行任何任务?无论哪种方式,Executor / ExecutorService如何将正确的任务分配给正确的线程? 参考方案 …

java字节代码(已编译的语言,也称为目标代码)与机器代码(当前计算机的本机代码)之间有什么区别?我读过一些书,他们将字节码称为二进制指令,但我不知道为什么。 参考方案 字节码是独立于平台的,在Windows中运行的编译器编译的字节码仍将在linux / unix / mac中运行。机器代码是特定于平台的,如果在Windows x86中编译,则它将仅在Win…

有哪些替代继承的方法? java大神给出的解决方案 有效的Java:偏重于继承而不是继承。 (这实际上也来自“四人帮”)。他提出的理由是,如果扩展类未明确设计为继承,则继承会引起很多不正常的副作用。例如,对super.someMethod()的任何调用都可以引导您通过未知代码的意外路径。取而代之的是,持有对本来应该扩展的类的引用,然后委托给它。这是与Eric…

我想将BigInteger写入文件。做这个的最好方式是什么。当然,我想从输入流中读取(使用程序,而不是人工)。我必须使用ObjectOutputStream还是有更好的方法?目的是使用尽可能少的字节。谢谢马丁 参考方案 Java序列化(ObjectOutputStream / ObjectInputStream)是将对象序列化为八位字节序列的一种通用方法。但…

我有一个使用org.restlet.engine.ssl.DefaultSslContextFactory的现有应用程序和一个在服务器启动时加载的密钥库文件。我有另一个应用程序,该应用程序创建必须添加的证书服务器运行时动态地更新到密钥库文件。为此,我在代码中创建了证书和私钥,然后将其写入到目录。该目录由bash脚本监视,该脚本检查是否有新文件,如果出现,它将…

赞(0) 打赏
部分文章转自网络,侵权联系删除b2bchain区块链学习技术社区 » 多个Java线程似乎锁定了同一监视器? – java程序员分享
分享到: 更多 (0)

评论 抢沙发

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址

b2b链

联系我们联系我们