这个小代码片段永远不会在 jdk8u45 上完成,并且曾经在 jdk8u20 上正确完成:
public class TestForkJoinPool {
final static ExecutorService pool = Executors.newWorkStealingPool(8);
private static volatile long consumedCPU = System.nanoTime();
public static void main(String[] args) throws InterruptedException {
final int numParties = 100;
final Phaser p = new Phaser(1);
final Runnable r = () -> {
p.register();
p.arriveAndAwaitAdvance();
p.arriveAndDeregister();
};
for (int i = 0; i < numParties; ++i) {
consumeCPU(1000000);
pool.submit(r);
}
while (p.getArrivedParties() != numParties) {}
}
static void consumeCPU(long tokens) {
// Taken from JMH blackhole
long t = consumedCPU;
for (long i = tokens; i > 0; i--) {
t += (t * 0x5DEECE66DL + 0xBL + i) & (0xFFFFFFFFFFFFL);
}
if (t == 42) {
consumedCPU += t;
}
}
}
Phaser的文档指出
在 ForkJoinPool 中执行的任务也可以使用 Phasers,这将确保在其他人被阻塞等待阶段推进时执行任务有足够的并行性。
但是ForkjoinPool#mangedBlock 的 javadoc指出:
如果在 ForkJoinPool 中运行,可能会首先扩展池以确保足够的并行度
那里只有一个可能。所以我不确定这是否是一个错误,或者只是不依赖 Phaser/ForkJoinPool 合约的错误代码:Phaser/ForkJoinPool 组合的合约有多难防止死锁?
我的配置:
- Linux adc 3.14.27-100.fc19.x86_64 #1 SMP 2014 年 12 月 17 日星期三 19:36:34 UTC x86_64 x86_64 x86_64 GNU/Linux
- 8核i7