1

我正在尝试学习 java 并发 API,对于我的练习,我想安排一个作业每隔 X 秒定期运行一次。该作业将计算一个随机数。

我想在计划任务完成后立即获得结果。我无法仅使用 API 完成此操作,所以我破解了它。

有没有办法在不使用低级机制的情况下做得更好?我希望能够删除同步MyRandomGiverTask.getResult(),而是使用类似ScheduledFuture.get(). 但在我的代码中,ScheduledFuture 从未完成/完成。这是我目前的解决方案:

class A {
    public static void main() {
        MyRandomGiverTask task = new MyRandomGiverTask(200);
        ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
        ScheduledFuture<Double> scheduledDouble =
            (ScheduledFuture<Double>) scheduler
                .scheduleAtFixedRate(task, 1, 4, TimeUnit.SECONDS);
        while (true) {
            System.out.println(" >> " + task.getResult());
        }
    }
    public class MyRandomGiverTask implements Runnable {
        MyRandomGiver giver = new MyRandomGiver();
        int param;
        double result;
        public MyRandomGiverTask(int param) { this.param = param; }
        @Override public void run() { result = giver.getRandom(param); }
        public double getResult() {
            try {
                while (result == 0d) {
                    try {
                        Thread.sleep(500);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                return result;
            } finally {
                result = 0d;
            }
        }
    }
}
4

2 回答 2

7

您的任务是按固定速率安排的。这意味着在您取消任务之前,它将由执行者以固定的速率一次又一次地执行。这样的任务唯一能做的就是产生副作用。它不能返回任何东西,因为执行器返回的未来代表了任务的所有未决执行。顺便说一句,您会注意到该schedule方法将 Callable 作为参数(可能会产生某些结果),而该sceduleAtFixedRate方法仅将 Runnable 作为参数(返回 void,因此无法返回任何内容)。

所以,如果你想打印每次执行的结果,那么只需让任务本身(Runnable)打印它的结果,或者让 runnable 把它的结果放在一个阻塞队列中,然后让主线程take从队列中取出。因此,主线程将被阻塞,直到将一些结果放入队列中。

于 2012-01-15T13:30:35.503 回答
2

如果要计算每个随机数,请使用 BlockingQueue。计划任务put()在队列中的新随机数以及任何想要它们的东西都可以take()

此外,如果您要使用类似您解决方案的东西,您会想要使用wait()/ notify(),而不是sleep()

于 2012-01-15T13:30:47.343 回答