Java初学者项目实战:开发任务调度与提醒系统

初学者应从 java.util.Timer + TimerTask 入手任务调度系统,因其异常直观、线程安全、取消简单;搭配 Swing 时需复用 Timer 实例、用 SwingUtilities.invokeLater 更新 UI,并用 DateTimeFormatter 解析时间、Properties 持久化任务。

Java初学者做任务调度与提醒系统,别从 Quartz 或 Spring Scheduler 入手——它们抽象层太厚,出错时连线程在哪崩的都找不到。真正适合练手的是 java.util.Timer + TimerTask 搭配基础 Swing/AWT 界面,先跑通“添加任务→延时执行→弹窗提醒”闭环,再逐步替换为更健壮的方案。

为什么 TimerTask 比 ScheduledExecutorService 更适合起步

初学者常被推荐用 ScheduledExecutorService,但它的 scheduleAtFixedRatescheduleWithFixedDelay 行为差异容易混淆,且异常不抛到主线程,调试时任务静默失败却无报错。而 TimerTaskrun() 方法在异常时会直接打印堆栈,你能立刻看到 NullPointerException 是因为 reminderLabel 还没初始化。

  • Timer 是单线程调度器,所有任务串行执行,不会出现并发修改 UI 组件(如 JLabel)导致的 IllegalStateException
  • 任务取消只需调用 timer.cancel(),不用管理 ScheduledFuture 引用
  • 构造 TimerTask 时可直接捕获外部变量(如 String taskName),无需写额外包装类

Swing 界面中触发定时任务的正确姿势

在按钮点击事件里直接 new Timer() 是常见错误——每次点“添加”就启一个新线程,旧任务无法取消,内存泄漏+弹窗叠罗汉。必须把 Timer 实例作为类成员,并在添加前检查是否已存在运行中的任务。

private Timer activeTimer;

private void scheduleReminder(String msg, long delayMs) {
    if (activeTimer != null) {
        activeTimer.cancel(); // 必须先取消旧 timer
    }
    activeTimer = new Timer();
    activeTimer.schedule(new TimerTask() {
        @Override
        public void run() {
            SwingUtilities.invokeLater(() -> {
                JOptionPane.showMessageDialog(null, "⏰ 提醒:" + msg);
                activeTimer = null; // 执行完清空引用,避免重复触发
            });
        }
    }, delayMs);
}

注意:SwingUtilities.invokeLater 不可省略,否则 JOptionPane 在非 EDT 线程调用会抛 IllegalArgumentException

时间输入解析:别硬写正则,用 DateTimeFormatter

用户输入 “5分钟”、“2小时后”、“今天14:30”,自己切字符串+判断单位极易出错。Java 8+ 直接用 DateTimeFormatterPeriod 更可靠:

  • 固定格式如 "HH:mm" → 用 LocalTime.parse(input, formatter) 得到今日该时刻
  • 相对描述如 "5m" → 正则提取数字和单位,用 Period.ofMinutes(n) 转换
  • 绝对日期如 "2025-03-15 10:00" → 用 LocalDateTime.parse(input, formatter)

关键点:所有解析结果统一转成 Instant,再减去 System.currentTimeMillis() 得到毫秒延迟值传给 timer.schedule()

任务持久化?先用 Properties 文件顶着

初学阶段别碰数据库或

JSON 库。java.util.Properties 足够存几条任务:

Properties props = new Properties();
props.setProperty("task1.time", "1710518400000"); // 毫秒时间戳
props.setProperty("task1.msg", "交作业");
props.store(new FileOutputStream("reminders.properties"), null);

启动时加载:props.load(new FileInputStream("reminders.properties")),遍历 key 匹配 "task.*.time",重建 TimerTask。注意文件路径用 new File(".").getAbsolutePath() 查看当前工作目录,避免因 IDE 启动路径不同导致找不到文件。

真正卡住初学者的不是语法,是搞不清 Timer 生命周期和 Swing 线程模型的耦合点——比如忘记 cancel 就关窗口,后台线程还在跑;或者在 TimerTask 里直接 setText,结果 UI 冻结。把这些链路亲手断一次、修一次,比抄十遍 Spring Boot 配置有用得多。