ScheduledExecutorService是Java中实现定时任务的首选方案,它比Timer更灵活且线程安全。通过Executors.newScheduledThreadPool创建线程池后,可使用scheduleAtFixedRate按固定频率执行任务,或用scheduleWithFixedDelay在任务完成后延迟指定时间再执行;二者区别在于调度时机:前者以固定周期为间隔,后者以上一任务结束为起点。使用时需注意捕获任务异常、合理配置线程池大小,并在程序退出前调用shutdown关闭线程池;对于复杂场景如Cron表达式支持,推荐Quartz或Spring Scheduling。在Spring中结合@Scheduled注解和@EnableScheduling可简化开发。

在Java中实现定时任务,ScheduledExecutorService 是最推荐的方式之一。它属于java.util.concurrent包,相比传统的Timer类,更加灵活、线程安全,并支持多任务调度和线程池管理。
1. 使用ScheduledExecutorService创建定时任务
通过Executors工厂类获取ScheduledExecutorService实例,然后调用其调度方法来安排任务的执行。
基本使用步骤:
- 使用Executors.newScheduledThreadPool(int corePoolSize)创建线程池。
- 调用scheduleAtFixedRate()或scheduleWithFixedDelay()方法设置定时逻辑。
- 传入Runnable任务、初始延迟、周期和时间单位。
- 任务执行完毕后,记得调用shutdown()关闭线程池。
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
public class ScheduledTaskExample {
public static void main(String[] args) {
// 创建一个大小为2的调度线程池
ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(2);
// 提交定时任务
scheduler.scheduleAtFixedRate(() -> {
System.out.println("执行任务: " + System.currentTimeMillis());
}, 1, 2, TimeUnit.SECONDS); // 初始延迟1秒,之后每2秒执行一次
// 模拟程序运行一段时间
try {
TimeUnit.SECONDS.sleep(10);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
// 关闭线程池
scheduler.shutdown();
}
}
2. scheduleAtFixedRate 与 scheduleWithFixedDelay 的区别
这两个方法都用于周期性执行任务,但行为不同:
立即学习“Java免费学习笔记(深入)”;
- scheduleAtFixedRate:按固定频率执行,例如每2秒一次,不管上个任务是否完成。如果任务执行时间超过周期,下一个任务会等它结束后立即开始(不会并发)。
- scheduleWithFixedDelay:在上一个任务完成后,延迟指定时间再执行下一次。适合任务耗时不固定,需要确保间隔的情况。
scheduler.scheduleWithFixedDelay(() -> {
long start = System.currentTimeMillis();
System.out.println("开始任务: " + start);
try {
Thread.sleep(3000); // 模拟耗时操作
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
System.out.println("任务结束: " + System.currentTimeMillis());
}, 1, 2, TimeUnit.SECONDS);
// 第一次延迟1秒,之后每次任务结束后等待2秒再启动下一次
3. 注意事项与最佳实践
使用ScheduledExecutorService时,有几个关键点需要注意:
- 避免任务中抛出未捕获异常,否则该任务将停止调度。建议在Runnable中使用try-catch包裹业务逻辑。
- 合理设置线程池大小,避免资源浪费或调度阻塞。
- 生产环境中应妥善处理shutdown,可结合Spring的@PreDestroy或JVM关闭钩子。
- 对于更复杂的调度需求(如Cron表达式),建议使用Quartz或Spring Scheduling。
4. 结合Spring使用(可选扩展)
在Spring项目中,可以通过@Scheduled注解简化定时任务开发:
@Scheduled(fixedRate = 2000)
public void doTask() {
System.out.println("Spring定时任务执行");
}
需在配置类上启用定时支持:@EnableScheduling。
基本上就这些。ScheduledExecutorService简单高效,是Java原生实现定时任务的首选方案。










