取消定时器线程后仍然在运行。

3
我不知道为什么线程在我取消它后仍然存在? 在下面的代码中,Timer线程实现了runnable接口,我想要它在使用Thread.sleep(7000);延迟7秒并在此之后调用mTimer.cancel();方法终止线程。在这7秒钟内,run()方法应该被调用以执行一个将持续10秒的任务。当我执行该程序时,第一行的时间是12:18:37,正如我所预期的那样,因为在我取消线程之前,该线程只会存活7秒钟,但出乎意料的是,在第二行显示的时间是05 12:18:44,现在这是7秒钟的差异,这是合理的。但令人意外的是,在3秒钟后控制台上又出现了一条记录,时间为05 12:18:47。如果你注意到第一行和第三行之间的差异,你会发现是10秒钟。我的问题是,为什么整个线程没有在7秒钟后停止。

Java 代码:

public class TimerTaskExample extends TimerTask {
private static Timer mTimer;

@Override
public void run() {
    // TODO Auto-generated method stub
    System.out.println("going to sleep at: " + new Date());
    dosomeWorks();
    System.out.println("returned back at: " + new Date());
}

private void dosomeWorks() {
    // TODO Auto-generated method stub
    try {
        Thread.sleep(10000);
    } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

public static void main(String args[]) {
    TimerTask timerTask = new TimerTaskExample();
    mTimer = new Timer(true);
    mTimer.scheduleAtFixedRate(timerTask, new Date(), 2*1000);
    //System.out.println("timer will start now at: " + new Date());
    try {
        Thread.sleep(7000);
    } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    mTimer.cancel();
     System.out.println("timer is cancelled now at: " + new Date());
     try {
        Thread.sleep(3000);
            } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

输出:

going to sleep at: Mon May 05 12:18:37 CEST 2014
timer is cancelled now at: Mon May 05 12:18:44 CEST 2014
returned back at: Mon May 05 12:18:47 CEST 2014

逻辑上的结论是,您无法立即取消正在休眠的线程。 - Squonk
1个回答

2
取消 TimerTask 不会中断其执行 - 它只是防止再次执行。
如果要停止线程执行,需要将 java.lang.Thread 的引用存储在某个地方并中断它。
假设您正在线程中执行实际工作,还需要轮询 Thread.interrupted() 来检查是否应该继续。
private Thread taskThread;

@Override
public void run() {
    taskThread = Thread.currentThread();
    System.out.println("going to sleep at: " + new Date());
    doSomeWorks();
    System.out.println("returned back at: " + new Date());
}

private void dosomeWorks() {
    for (int i = 0; !Thread.interrupted() && i < 10; ++i) {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

...
mTimer.cancel();
timerTask.taskThread.interrupt();
...

一定要阅读Thread.interrupt()的javadoc,以确保您了解它实际上是做什么的。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接