4

我对 Java 中 Timer 类的行为有疑问。这是代码: http: //pastebin.com/mqcL9b1n

public class Main {

    public static void main(String[] args) {
        Main m = new Main();
        m.foo();
        m = null;
    }

    public void foo() {
        Timer t = new Timer();
        t.schedule(new SysPrint(), 200);
    }

}

class SysPrint extends TimerTask {

    public void run() {
        System.out.println("Yes!");
    }
}

发生的情况是,如果您运行该程序,它将打印“是!” 它不会做任何其他事情(程序不会结束)。

Java 文档说:在对 Timer 对象的最后一个实时引用消失并且所有未完成的任务都完成执行后,计时器的任务执行线程优雅地终止(并成为垃圾收集的对象)。

正如我看到的那样,在“foo()”函数结束后,对 Timer 对象的“最后一次实时引用”就消失了。唯一安排的任务是“是!” 执行的任务,所以我猜在进程打印“是!”之后,Timer 对象应该结束并且进程应该终止。

这里发生了什么?

4

3 回答 3

2

Java 没有退出,因为运行 Timer 的线程仍在运行。在 Java 退出之前,您必须将该线程标记为守护线程。您可能无法访问线程本身,因此除非 Timer 有一种方法来标记它,否则您将很难做到这一点。您需要在 finally 子句中手动停止它。

try {
   timer = new Timer();
   timer.schedule( new SysPrint(), 200 );
} finally {
   timer.cancel();
}
于 2010-08-17T03:12:34.580 回答
0

我相信下面的代码应该可以解决问题。

public class Main {
  public static void main(String[] args) {
    Main m = new Main();
    m.foo();
    m = null;
  }

  public void foo() {
    Timer t = new Timer();
    t.schedule(new SysPrint(), 200);
  }
}

class SysPrint extends TimerTask {
  SysPrint(Timer timer) {
    this.timer = timer;
  }

  public void run() {
    System.out.println("Yes!");
    timer.cancel();
  }

  private Timer timer;
}
于 2019-11-23T17:17:32.987 回答
0

当你创建一个 Timer 对象时。创建了一个 TimerThread。它是运行您的任务的内部线程。可以查看TimerThread的run()方法。你可以看到它有一个while循环。

 private void mainLoop() {
    while (true) {....

TimerThread 没有设置为守护进程,所以 main 方法完全执行,jvm 不存在。

这就是为什么您的程序始终在运行并且不会停止的原因。

于 2020-03-14T08:40:58.503 回答