应使用 bufferedreader 包裹 system.in 进行控制台输入,因其 readline() 总读完整行、行为确定;写日记用 files.write() 配合 append/create 和 utf-8 编码;读日记用 files.readalllines() 但需捕获 nosuchfileexception 并判空;路径应封装为静态常量并优先使用用户主目录。

用 BufferedReader + System.in 做控制台输入,别直接用 Scanner
控制台交互时,Scanner 遇到换行或空格容易吃掉后续输入,尤其在读完一行再读日期、标题时经常丢数据。用 BufferedReader 包裹 System.in 更可控。
-
Scanner的nextLine()和nextInt()混用会残留换行符,导致下一次nextLine()直接返回空字符串 -
BufferedReader.readLine()总是读完整行,行为确定,适合日记这种“一行一字段”的场景 - 记得在
try-with-resources里声明,避免System.in被意外关闭(虽然它关不掉,但写法要一致)
示例:获取日记标题
BufferedReader console = new BufferedReader(new InputStreamReader(System.in));
System.out.print("请输入标题:");
String title = console.readLine().trim();
写日记用 Files.write() 追加模式,不是 FileWriter
日记本的核心是“追加”,不是覆盖。用 FileWriter 开启 true 参数看似可行,但它不处理换行、编码、路径不存在等问题;而 Files.write() 是 Java 7+ 推荐的现代方式,天然支持 UTF-8 和自动创建父目录。
- 必须显式传入
StandardOpenOption.APPEND,否则默认覆盖 - 一定要加
StandardOpenOption.CREATE,否则文件不存在时抛NoSuchFileException - 别漏掉
Charset.forName("UTF-8"),Windows 控制台默认 GBK,不指定会导致中文乱码 - 每条日记建议以换行开头(即先写
"\n"),避免首行粘连或空行缺失
示例:追加一条格式化日记
立即学习“Java免费学习笔记(深入)”;
String entry = "\n[" + new Date() + "] " + title + "\n" + content;
Files.write(Paths.get("diary.txt"), entry.getBytes(StandardCharsets.UTF_8),
StandardOpenOption.CREATE, StandardOpenOption.APPEND);
读日记用 Files.readAllLines() 要小心内存和空文件
显示历史日记时,想一次性加载全部内容最方便,但 Files.readAllLines() 会把整个文件读进内存。对日记本来说一般没问题,但得防两种边界情况:文件不存在、文件为空。
- 文件不存在时抛
NoSuchFileException,必须捕获,不能假设它一定存在 - 空文件返回空
List<string></string>,不是null,所以判空要用list.isEmpty(),不是list == null - 如果日记量极大(比如上万条),改用
Files.lines()流式处理,但日常使用没必要 - 注意行尾换行符已被剥离,显示时需手动补
\n或用System.lineSeparator()
示例:安全读取并打印
try {
List<String> lines = Files.readAllLines(Paths.get("diary.txt"), StandardCharsets.UTF_8);
if (lines.isEmpty()) {
System.out.println("暂无日记");
} else {
lines.forEach(System.out::println);
}
} catch (NoSuchFileException e) {
System.out.println("日记文件不存在,尚未写入任何内容");
}
路径硬编码成 "diary.txt" 很危险,至少封装成常量
开发时图省事写死路径,上线后可能因工作目录变化导致文件写到奇怪位置,甚至权限失败。Java 没有全局配置概念,但一个静态常量比到处写字符串强得多。
- 路径尽量用相对路径,但确保它相对于用户期望位置(比如用户主目录:
System.getProperty("user.home") + "/diary.txt") - 如果用当前目录,确认 IDE 或 jar 启动时的工作目录是否可控(IntelliJ 默认是项目根,jar 默认是执行目录)
- 不要用
File.separator拼接路径,一律用Paths.get(...),它自动适配 Windows/Linux - 路径中含中文时,
Paths.get()没问题,但某些旧终端或打包工具可能解析异常,测试时务必实测
示例:定义可维护路径
private static final Path DIARY_PATH = Paths.get(System.getProperty("user.home"), "my-diary.txt");
实际写的时候,最容易被忽略的是编码一致性——控制台输入、文件写入、文件读取三处都得明确指定 UTF_8,少一处就可能看到一堆问号或乱码。还有就是空文件和首次运行的异常分支,很多人只测了“已经写了好几条”的情况。









