推荐使用ScheduledExecutorService实现定时任务,因其支持多线程、异常隔离和更精确调度,而Timer单线程且异常会终止全部任务,存在缺陷。

在Java中实现定时任务,常用的方式有两种:使用 Timer 类和使用 ScheduledExecutorService。虽然两者都能完成定时或周期性任务的调度,但推荐使用 ScheduledExecutorService,因为它更健壮、更灵活,且线程安全性更好。
1. 使用 Timer 实现定时任务
Timer 是 Java 早期提供的定时器工具,通过一个后台线程按计划执行任务。它适合简单的定时场景,但存在一些局限性(如异常会终止整个定时器)。
示例:每隔2秒执行一次任务
import java.util.Timer;
import java.util.TimerTask;
public class TimerExample {
public static void main(String[] args) {
Timer timer = new Timer();
TimerTask task = new TimerTask() {
@Override
public void run() {
System.out.println("执行定时任务: " + System.currentTimeMillis());
}
};
// 延迟1秒后开始,每隔2秒执行一次
timer.schedule(task, 1000, 2000);
}
}
注意:如果 TimerTask 中抛出未捕获的异常,整个 Timer 的后续任务都会停止,这是其主要缺陷。
2. 使用 ScheduledExecutorService 实现定时任务
ScheduledExecutorService 是 java.util.concurrent 包中的接口,比 Timer 更强大,支持多线程调度、异常隔离、更精确的时间控制等。
立即学习“Java免费学习笔记(深入)”;
示例:延迟1秒后,每隔2秒执行一次
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
public class ScheduledExecutorExample {
public static void main(String[] args) {
ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(2);
Runnable task = () -> System.out.println("执行任务: " + System.currentTimeMillis());
// 延迟1秒后开始,每隔2秒执行一次
scheduler.scheduleAtFixedRate(task, 1, 2, TimeUnit.SECONDS);
}
}
常用方法说明:
- schedule(Runnable command, long delay, TimeUnit unit):延迟执行一次
- scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit unit):固定频率执行,从任务开始计时
- scheduleWithFixedDelay(Runnable command, long initialDelay, long delay, TimeUnit unit):固定延迟执行,从上一次结束开始计时
3. Timer 和 ScheduledExecutorService 对比
| 特性 | Timer | ScheduledExecutorService |
|---|---|---|
| 线程模型 | 单线程 | 可配置多线程 |
| 异常处理 | 一个任务异常会导致其他任务失效 | 任务间隔离,异常不影响其他任务 |
| 精度与性能 | 一般 | 更高,适合复杂调度 |
| 推荐程度 | 不推荐用于生产环境 | 推荐,是现代Java应用首选 |
4. 实际使用建议
在实际开发中,特别是Spring等框架中,更多使用 @Scheduled 注解来实现定时任务,底层通常基于 ScheduledExecutorService 或 Quartz。
但如果需要手动控制线程池或调度逻辑,直接使用 ScheduledExecutorService 更加灵活可靠。
基本上就这些,ScheduledExecutorService 更安全、更强大,应优先选择。Timer 只适合学习或非常简单的场景。











