ScheduledExecutorService 是实现周期性提醒任务最稳妥的 Java 标准库方案,线程安全、精度高、可取消、异常不中断调度,配合 ConcurrentHashMap 与 TriggerPolicy 接口可支撑多模式、高并发、低延迟提醒。

用 ScheduledExecutorService 启动周期性提醒任务最稳妥
Java 标准库中,ScheduledExecutorService 是实现轻量级定时提醒的首选——它不依赖 Spring,线程安全,且能精确控制执行延迟和间隔。相比老旧的 Timer,它不会因单个任务异常而终止整个调度器。
常见错误是直接 new Thread + sleep():不仅精度差(受 GC 和系统调度影响),还无法取消、无法复用、无法捕获异常。
- 用
Executors.newScheduledThreadPool(1)创建单线程池,避免多任务并发干扰提醒逻辑 - 调用
scheduleAtFixedRate()适合「每 N 秒执行一次」场景(如每 30 秒检查一次待提醒事项) - 若任务执行时间可能超过间隔,改用
scheduleWithFixedDelay(),它在上一次执行**结束后**再等指定延迟 - 务必保存返回的
ScheduledFuture,后续可用cancel(true)中断正在运行的任务
ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
ScheduledFuture> reminderTask = scheduler.scheduleAtFixedRate(() -> {
List due = findDueReminders(); // 自定义查询逻辑
due.forEach(r -> System.out.println("? 提醒:" + r.getContent()));
}, 0, 30, TimeUnit.SECONDS); 提醒数据怎么存?内存 Map 就够用,别一上来就上数据库
简易系统的核心是「快启动、易调试、无外部依赖」。用 ConcurrentHashMap 存 Reminder 对象,配合毫秒级时间戳判断是否到期,比连 MySQL 或 Redis 快一个数量级,也避免连接泄漏和事务干扰。
容易踩的坑:用 HashMap 替代 ConcurrentHashMap,多线程下可能死循环;或把 System.currentTimeMillis() 写死在构造里,导致创建后时间永远不变。
立即学习“Java免费学习笔记(深入)”;
-
Reminder至少包含id、content、triggerTime(long 类型毫秒时间戳) - 查询过期项时,用
map.values().stream().filter(r -> r.triggerTime - 清理已触发项必须加同步块或用
computeIfPresent原子操作,否则可能漏删或重复提醒
如何让提醒支持「只触发一次」和「每天重复」两种模式?
硬编码 if-else 判断类型会迅速变复杂。更清晰的做法是抽象出 TriggerPolicy 接口,定义 nextTriggerTime(long lastTime) 方法:
- 「只触发一次」实现直接返回 -1(表示不再调度)
- 「每天重复」实现可返回
lastTime + 24 * 60 * 60 * 1000,但要注意跨天时区问题——建议统一用LocalDateTime.now().plusDays(1).atZone(ZoneId.systemDefault()).toInstant().toEpochMilli() - 任务触发后,先调用
policy.nextTriggerTime(current),若结果 > 0,则更新该Reminder的triggerTime并保留;否则从 Map 中移除
这样新增「每周二上午 9 点」策略,只需新增一个实现类,原调度逻辑完全不动。
控制台输出不够?加个 Desktop.getDesktop().browse() 弹窗提醒
纯 console 输出容易被忽略,尤其当 IDE 或终端不在焦点时。Java 7+ 提供 Desktop API,可在 macOS / Windows / Linux 上唤起默认浏览器或播放声音。
注意:Linux 下需安装 xdg-utils,Windows 需确保 java.awt.Desktop.isDesktopSupported() 和 .isSupported(Desktop.Action.BROWSE) 均为 true,否则会抛 UnsupportedOperationException。
- 弹窗提示推荐用
Desktop.getDesktop().beep()—— 轻量、无权限问题、所有平台都支持 - 想打开网页通知页?用
Desktop.getDesktop().browse(URI.create("http://localhost:8080/notify?id=" + id)) - 绝对不要在定时任务线程里调用
SwingUtilities.invokeLater()显示 JFrame:没设置setDefaultCloseOperation容易卡住 JVM 退出
真正难的不是写完第一个提醒,而是当用户添加了 200 条不同周期的提醒后,调度器仍保持低延迟、不丢任务、不累积延迟——这要求你从第一天就用 ScheduledExecutorService + 原子状态更新 + 明确的过期清理路径,而不是靠 sleep 和轮询硬扛。










