Java 计时器()线程在取消后仍在运行

Java 计时器()线程在取消后仍在运行,java,android,multithreading,timer,Java,Android,Multithreading,Timer,我不知道为什么在我取消线程后它仍然是活动的?在下面的代码中,定时器线程实现了可运行,在代码中,我希望它使用线程保持运行7秒。sleep(7000)之后,我调用了mTimer.cancel()终止线程。在7秒内,应调用run()方法来执行持续10秒的任务。当我执行该程序时,第一行的时间是12:18:37,正如我所预期的,因为在我取消它之前,该线程将只运行7秒,但是第二行显示的时间是05 12:18:44,现在这是7秒的差异,这是合理的。但出乎意料的是,3秒钟后,另一行出现在控制台上,并被计时到05

我不知道为什么在我取消线程后它仍然是活动的?在下面的代码中,
定时器
线程实现了
可运行
,在代码中,我希望它使用
线程保持运行7秒。sleep(7000)之后,我调用了
mTimer.cancel()
终止线程。在7秒内,应调用
run()
方法来执行持续10秒的任务。当我执行该程序时,第一行的时间是
12:18:37
,正如我所预期的,因为在我取消它之前,该线程将只运行7秒,但是第二行显示的时间是
05 12:18:44
,现在这是7秒的差异,这是合理的。但出乎意料的是,3秒钟后,另一行出现在控制台上,并被计时到
0512: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

取消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();
...
私有线程taskThread;
@凌驾
公开募捐{
taskThread=Thread.currentThread();
System.out.println(“将在:+new Date()睡觉);
doSomeWorks();
System.out.println(“返回时间:“+new Date()”);
}
私人工程{
对于(int i=0;!Thread.interrupted()&&i<10;++i){
试一试{
睡眠(1000);
}捕捉(中断异常e){
e、 printStackTrace();
}
}
}
...
mTimer.cancel();
timerTask.taskThread.interrupt();
...

请务必阅读javadoc,以确保您知道它确实如此。

逻辑结论是您不能立即取消休眠线程。