我正在尝试使用下面的两个线程来完成它。有人能指出我在这里做的明显错误吗?
public class OddEven {
public static boolean available = false;
public static Queue<Integer> queue = new LinkedList<Integer>();
static Thread threadEven = new Thread() {
@Override
public void run() {
printEven();
}
public synchronized void printEven() {
while (!available) {
try {
wait();
Thread.sleep(2000);
} catch (InterruptedException e) {
}
}
System.out.println(queue.remove());
available = false;
notifyAll();
}
};
static Thread threadOdd = new Thread() {
@Override
public void run() {
printOdd();
}
public synchronized void printOdd () {
while (available) {
try {
wait();
Thread.sleep(2000);
} catch (InterruptedException e) {
}
}
System.out.println(queue.remove());
available = true;
notifyAll();
}
};
public static void main(String[] args) {
int n = 20;
for (int i = 1; i < n; i++) {
queue.add(i);
}
threadOdd.start();
threadEven.start();
try {
Thread.sleep(60000);
} catch (InterruptedException e) {
e.printStackTrace();
}
try {
threadOdd.join();
threadEven.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
但是这个程序只打印1并退出。打印1后,available应为true,printEven应该唤醒,打印并设置为false。我不明白这里出了什么问题?我看到了其他解决方案,但想知道为什么我的解决方案无效。
答案 0 :(得分:1)
对实例方法进行同步意味着调用该方法的线程必须获取该实例的锁定; <{1}}是
的语法糖public void synchronized printOdd()
其中public void printOdd() {
synchronized(this) {
...
}
}
对于每个实例都不同,因为ThreadOdd和threadEven是两个不同的对象,每个对象都使用自己的锁。在用作锁的对象上调用notifyAll和wait方法。当一个线程等待时,它永远不会得到通知,因为该通知仅适用于与通知线程在同一个锁上等待的其他线程。