试图停止线程,在EnergyTransferTask类中加入退出旗标,修改后代码如下:
public class EnergyTransferTask implements Runnable { private EnergySystem energySystem;// 共享的能量世界 private int fromBox;// 能量转移的源能量盒子下标 private double maxAmount;// 单次能量转移最大单元 private int DELAY = 10;// 最大休眠时间(毫秒) volatile boolean keepRunning = true; private int count = 0; public EnergyTransferTask(EnergySystem energySystem, int from, double max) { this.energySystem = energySystem; this.fromBox = from; this.maxAmount = max; } public void run() { try { while (keepRunning) { int toBox = (int) (energySystem.getBoxAmount() * Math.random()); double amount = maxAmount * Math.random(); energySystem.transfer(fromBox, toBox, amount); Thread.sleep((int) (DELAY * Math.random())); count++; if(count == 5){ keepRunning = false; } } } catch (InterruptedException e) { e.printStackTrace(); } } }
为了便于研究,将盒子数量BOX_AMOUNT改为2,但发现控制台的输出停止后,程序仍无法正常退出。推测是在wait set中线程一直在等待中,而其他线程已经执行完毕无法将其唤醒,有什么好的解决方法吗?
没看到你别的代码,也没看这个题目,不知道具体问题, 但是就你出现的那个问题,notify是唤醒线程池中的任意一个线程,当你的读和取都有多个线程时,有使用的同一个锁,确实会造成死锁,
JDK1.5 java.util.concurrent.locks 包中提供了更方便的灵活的解决办法
java.util.concurrent.locks包下
Lock接口:代替了同步代码块或者同步函数,将同步的隐式锁操作变为了显示操作,而已可以加上多个监视器,
Condition接口中
signal();唤醒锁上指定监视器的一个线程 代替了notify()方法不会造成死锁
十分感谢楼上的同学指明方向,将synchronized同步机制改成ReentrantLock的lock机制,可以解决这个问题。
因此在EnergySystem类中声明:
private final ReentrantLock lock = new ReentrantLock(); private Condition notEnouge = lock.newCondition(); private Condition enouge = lock.newCondition();
并将该类中的synchronized块改为:
lock.lock(); try { while (energyBoxes[from] < amount) { notEnouge.await(1, TimeUnit.SECONDS);// wait(); } System.out.print(Thread.currentThread().getName()); energyBoxes[from] -= amount; System.out.printf("从%d转移%10.2f单位能量到%d", from, amount, to); energyBoxes[to] += amount; System.out.printf(" 能量总和:%10.2f%n", getTotalEnergies()); enouge.signalAll();// notifyAll(); } catch (InterruptedException e) { e.printStackTrace(); } finally { lock.unlock(); }
这样,等待中的线程每隔1秒就可重新争取锁,就不会陷入无限等待中。
发现有人这么说
wait notify 可以被认为是过时的机制,自从concurrent包之后,不要再用 wait notify 了,所以研究的意义可能也不大……要一个线程去等另一个线程,或者等某一个条件达成,concurrent包里有不好工具可以用,方法也不止一种(看情况用哪种),每一种都比 wait notify 好。