gpt4 book ai didi

java - 即使正在释放许可,Semaphore 类是否会进入死锁?

转载 作者:行者123 更新时间:2023-11-30 07:25:55 26 4
gpt4 key购买 nike

以下使用 Java 的 Semaphore 类制作的并发代码进入了死锁,甚至很艰难,根据控制台输出,许可证正在释放。

package ThreadTraining;

import java.util.concurrent.Semaphore;

public class ThreadTraining {

public static class Value {

private static int value = 0;
private static final Semaphore SEMAPHORE = new Semaphore(1);

public static synchronized void acquire() throws InterruptedException {
SEMAPHORE.acquire();
System.out.println("A thread has aquired a permit!");
}

public static synchronized void release() {
SEMAPHORE.release();
}

public static int get() {
return value;
}

public static void add() {
value++;
}

public static void subtract() {
value--;
}
}

public static class Adder extends Thread {

public Adder(String name) {
this.setName(name);
}

@Override
public void run() {
System.out.println(this.getName() + " has been created.");
boolean keepRunning = true;
while (keepRunning) {
try {
Value.acquire();
System.out.print(this.getName() + " has aquired Value's permit. --- ");
if (Value.get() > 99) {
System.out.print(this.getName() + " has finished it's job. --- ");
keepRunning = false;
} else {
System.out.print(this.getName() + " has modified value from " + Value.get() + " to ");
Value.add();
System.out.println(Value.get() + ".");
}
} catch (InterruptedException ie) {
System.err.println("This thread was interrupted.");
} finally {
System.out.println(this.getName() + " is releasing Value's permit.");
Value.release();
}
}
}
}

public static void main(String[] args) {
Thread threads[] = new Thread[3];
for (int i = 0; i < threads.length; i++) {
threads[i] = new Adder("[Adder]Thread #" + i);
}
for (Thread t : threads) {
t.start();
}
}
}

代码的控制台输出:(这是一个“幸运的”运行,它通常只打印到指定的点)

[Adder]Thread #0 has been created.
[Adder]Thread #1 has been created.
[Adder]Thread #2 has been created.
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 0 to 1.
[Adder]Thread #0 is releasing Value's permit. /*NOTE: It usually prints only up to this line, hanging after the first permit-release.*/
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 1 to 2.
[Adder]Thread #0 is releasing Value's permit.
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 2 to 3.
[Adder]Thread #0 is releasing Value's permit.
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 3 to 4.
[Adder]Thread #0 is releasing Value's permit.
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 4 to 5.
[Adder]Thread #0 is releasing Value's permit.
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 5 to 6.
[Adder]Thread #0 is releasing Value's permit.
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 6 to 7.
[Adder]Thread #0 is releasing Value's permit.
A thread has aquired a permit!
[Adder]Thread #0 has aquired Value's permit. --- [Adder]Thread #0 has modified value from 7 to 8.
[Adder]Thread #0 is releasing Value's permit.

背后的原因是什么?并且,如果可能的话,如何修复它?


附加信息:

这个问题是my previous concurrency question的“延续” .

新代码非常基于this semaphore tutorial .

最佳答案

问题是你自己方法的同步问题。 Value.acquireValue.release 都是同步的,因此进入您的 acquire 方法的线程将阻止另一个线程调用 release ,因为 release 调用将等待 Value 类的监视器被释放,而 acquire 中的监视器将等待要获取的内部信号量。从您的方法中删除 synchronized -关键字,您将摆脱死锁问题。相反,您可能打算同步您的 get-、add- 和 subtract- 方法。

关于java - 即使正在释放许可,Semaphore 类是否会进入死锁?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10640362/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com