我有一个疑问:假定我有一个名为sem的信号量类的对象,并且某个线程尝试获取执行sem.acquire()的权限,并且没有,因此该线程被挂起。现在,如果线程被虚假唤醒唤醒,会发生什么?我应该在某种循环中插入sem.acquire()吗?

最佳答案

只有非常基本的Object.wait()方法会受到虚假唤醒(实际上只有Unsafe.park()方法,但是没有办法可以在没有黑客的情况下进行访问),所有更复杂的工具都会在内部进行处理。

如果您查看Semaphore.acquireUninterruptibly()的实际代码,则会发现以下构造(隐藏在AbstractQueuedSynchronizer.doAcquireShared()中):

for (;;) {
    // ... try aquire, else:
    if (shouldParkAfterFailedAcquire(p, node) &&
        parkAndCheckInterrupt())
        interrupted = true;
}


for(;;)循环基本上是一个while(true)循环,因此即使parkAndCheckInterrupt()可能会虚假地唤醒,该循环也将确保这只会导致另一个tryAquire运行失败,然后将其重新休眠。

10-07 19:16
查看更多