Java初学者项目实战:编写记账本应用程序

记账本是检验Java基础串联能力的试金石,核心在于Scanner输入处理(nextInt后需nextLine清换行符或统一用nextLine+parse)、ArrayList存Record对象、追加模式文件写入(FileWriter("f",true))、容错式数据加载(跳过空行/格式错误)及必要日志调试。

记账本不是练手玩具,而是检验你能否把 Java 基础串成可用逻辑的第一块试金石。它不考算法,但会暴露你对 Scanner 输入处理、ArrayList 动态存取、文件持久化(FileWriter/BufferedWriter)和异常边界(InputMismatchExceptionIOException)的真实掌握程度。

用 Scanner 读用户输入时,nextLine() 被 nextInt() 吃掉怎么办

这是新手写记账本最常卡住的点:先用 nextInt() 读金额,再用 nextLine() 读备注,结果备注永远为空。

  • nextInt() 不会消费换行符,光标停在换行符上;下一次 nextLine() 立刻读到空字符串
  • 正确做法是:在 nextInt() 后加一句 scanner.nextLine() 手动清掉残留换行符
  • 更稳妥的方案是统一用 nextLine(),再用 Integer.parseInt()Double.parseDouble() 转数字——虽然要自己捕获 NumberFormatException

记账数据该存在 ArrayList 还是 Map

单账户、按时间顺序增删查的场景,ArrayList 是最直接选择。别一上来就用 HashMap 模拟“键值对记账”——那是在给自己造类型混乱的坑。

  • Record 类至少应含 dateStringLocalDate)、amountdouble)、categoryString)、noteString
  • ArrayList 支持按索引删除(如删除第 3 笔)、遍历统计(如本月支出总

    和)、简单条件筛选(如查“餐饮”类目)
  • 如果后续要支持“按日期快速查找”,再考虑用 TreeMap>,而不是一开始就强行抽象

保存到文件时,覆盖写入还是追加写入

记账本要求每次新增记录都落盘,且不能丢失历史数据。用 new FileWriter("records.txt") 是覆盖模式,一重启程序,之前所有账目全丢。

  • 必须用 new FileWriter("records.txt", true) 第二个参数设为 true 开启追加
  • 每条记录建议用固定分隔符(如 |)拼接字段:2025-05-20|128.5|餐饮|外卖,方便后期用 split("\\|") 解析
  • 写完务必调用 bw.flush()bw.close(),否则缓冲区内容可能滞留不落地
try (BufferedWriter bw = new BufferedWriter(new FileWriter("records.txt", true))) {
    bw.write(record.getDate() + "|" + record.getAmount() + "|" + record.getCategory() + "|" + record.getNote());
    bw.newLine();
} catch (IOException e) {
    System.err.println("保存失败:" + e.getMessage());
}

启动时从文件加载数据,遇到空行或格式错误怎么跳过

用户可能手动编辑过 txt 文件,或上次写入中途崩溃,导致某行缺字段、多字段、金额非数字。硬抛异常会让整个程序起不来。

  • 逐行读 br.readLine(),对每行用 line.trim().isEmpty() 跳过空行
  • line.split("\\|", -1) 并检查数组长度是否 ≥ 4,不够就 continue
  • 解析金额时用 try-catch NumberFormatException,捕获后打印警告(如“第5行金额格式错误,已跳过”),不中断加载
  • 别把加载失败当作致命错误——记账本核心是能继续记新账,旧数据缺失只是部分损失

真正的难点不在功能实现,而在于你是否愿意花十分钟,在 saveToFile()loadFromFile() 里各加三行日志输出,去确认哪一行没写进去、哪一行被跳过了。没有这一步,你永远在猜 bug 在哪。