Java Timer

2022-09-02 13:09:28

我正在尝试使用计时器来计划在应用程序中定期发生事件。但是,我希望能够实时调整事件触发的时间段(根据用户输入)。

例如:

public class HelperTimer extends TimerTask
{
    private Timer timer;
    //Default of 15 second between updates
    private int secondsToDelay = 15;

    public void setPeriod(int seconds)
    {
        this.secondsToDelay = seconds;
        long delay = 1000; // 1 second
        long period = 1000*secondsToDelay; // seconds
        if (timer != null) 
        {
            timer.cancel();
        }
        System.out.println(timer);
        timer = new Timer();
        System.out.println(timer);
        timer.schedule(this, delay, period);
    }
    public int getPeriod()
    {
        return this.secondsToDelay;
    }
}

然后,我启动此类的新实例并调用其 set period 函数。但是,当我这样做时,我会收到一个非法状态异常。你可以看到 System.out.println(timer);在那里,因为我正在检查,是的,果然,它们是两个不同的计时器......那么,为什么当我尝试在全新的计时器实例上运行计划调用时,我会收到一个 IllegalStateException!?!?!?!

java.util.Timer@c55e36
java.util.Timer@9664a1
Exception in thread "AWT-EventQueue-0" java.lang.IllegalStateException: Task already scheduled or cancelled
    at java.util.Timer.sched(Unknown Source)
    at java.util.Timer.schedule(Unknown Source)
    at HelperTimer.setPeriod(HelperTimer.java:38)

答案 1

您不能像在这里所做的那样重用TimerTask。

定时器的相关孔径:

private void sched(TimerTask task, long time, long period) {
    if (time < 0)
        throw new IllegalArgumentException("Illegal execution time.");

    synchronized(queue) {
        if (!thread.newTasksMayBeScheduled)
            throw new IllegalStateException("Timer already cancelled.");

        synchronized(task.lock) {
            //Right here's your problem.
            //  state is package-private, declared in TimerTask
            if (task.state != TimerTask.VIRGIN)
                throw new IllegalStateException(
                    "Task already scheduled or cancelled");
            task.nextExecutionTime = time;
            task.period = period;
            task.state = TimerTask.SCHEDULED;
        }

        queue.add(task);
        if (queue.getMin() == task)
            queue.notify();
    }
}

您需要重构代码,以便创建新的 TimerTask,而不是重用一个计时器任务。


答案 2

对我来说,有一个带有自己的计时器的TimerTask似乎很奇怪。糟糕的设计。我将两者完全分开,并将 TimerTask 实现交给 Timer,并将所有关于摆弄句点的逻辑放在另一个类中,该类提供了执行此操作的接口。让该类实例化计时器和计时器任务,并让它们完成其工作。