我自己找到了一些答案。要测试当前线程是否拥有
监视器,是否 存在!
Thread.holdsLock
exists!
if (!Thread.holdsLock(data)) { throw new RuntimeException(); // complain}这确实非常快(亚微秒),并且从1.4开始就可用。
通常,要测试哪个线程(或线程ID)持有该锁,可以
对
java.lang.managementclasses (thanks @amicngh).
public static long getMonitorOwner(Object obj) { if (Thread.holdsLock(obj)) return Thread.currentThread().getId(); for (java.lang.management.ThreadInfo ti : java.lang.management.ManagementFactory.getThreadMXBean() .dumpAllThreads(true, false)) { for (java.lang.management.MonitorInfo mi : ti.getLockedMonitors()) { if (mi.getIdentityHashCode() == System.identityHashCode(obj)) { return ti.getThreadId(); } } } return 0;}有一些注意事项:
- 这有点慢(在我的情况下约为1/2毫秒,大概与线程数成线性关系)。
- 它需要Java 1.6和适用于它的
VM ThreadMXBean.isObjectMonitorUsageSupported()
,因此移植性较差。 - 它需要“监视器”安全权限,因此大概无法在沙盒小程序中使用。
- 如果需要的话,将线程ID转换为Thread对象并不是一件容易的事,因为我想您必须使用Thread.enumerate然后循环遍历以找出具有ID的对象,但这在理论上是有帮助的竞争条件,因为到您调用枚举时,该线程可能不再存在,或者可能出现了具有相同ID的新线程。
但是,如果您只想测试当前线程,Thread.holdsLock
效果
很好!否则,的实现 可能比普通的Java监视器提供更多的信息和灵活性 (感谢@ user1252434)。java.util.concurrent.locks.Lock



