DateTimeFormatter的核心作用是格式化时间对象为字符串及解析字符串为时间对象,是Java 8后线程安全的唯一推荐方案,替代了不安全的SimpleDateFormat。
DateTimeFormatter 的核心作用就两个:**把 LocalDate、LocalDateTime 等时间对象转成字符串(格式化)**,以及**把符合规则的字符串还原成时间对象(解析)**。它不是装饰用的工具类,而是 Java 8 之后处理时间 ↔ 字符串转换的**唯一推荐入口**——替代了线程不安全又难维护的 SimpleDateFormat。
这不是“升级可选”,而是实际工程中的硬性要求:
SimpleDateFormat 是可变对象,多线程共用一个实例时,parse() 或 format() 极易因内部状态混乱抛出 java.lang.NumberFormatException 或返回错误时间(比如把 "2025-12-30" 解析成 1970 年)DateTimeFormatter 是不可变(immutable)且线程安全的,定义一次可全局静态复用,无同步开销yyyy-MM-dd 去解析 "2025/12/30",会明确报错 Text '2025/12/30' could not be parsed at index 4,直接定位到斜杠位置别一上来就写 ofPattern —— 先看是否已有现成方案:
LocalDateTime.parse("2025-12-30T08:54:12", DateTimeFormatter.ISO_LOCAL_DATE_TIME)
ofLocalizedDateTime + Locale,避免手写 “年/月/日”DateTimeFormatter.ofLocalizedDateTime(FormatStyle.MEDIUM).withLocale(Locale.SIMPLIFIED_CHINESE)
ofPattern,注意大小写敏感:MM 是月,mm 是分;HH 是 24 小时制,hh 是 12 小时制DateTimeFormatter.ofPattern("yyyy年MM月dd日 HH:mm:ss")
看似简单,但 80% 的 DateTimeParseException 都源于这几点:
比如模式是 "yyyy-MM-dd",但字符串是 "2025-12-30 "(末尾有空格),会直接失败 —— 解析前建议先 trim()
ofPattern("yyyy年MM月dd日") 解析 "2025年12月30日" 时,若未显式传入 Locale.SIMPLIFIED_CHINESE,在某些 JVM 默认语言环境下可能识别不了“年/月/日”LocalDateTime.parse(...) 去解析只有日期的字符串(如 "2025-12-30"),会报错;应改用 LocalDate.parse(...) + DateTimeFormatter.ISO_LOCAL_DATE
LocalDateTime now = LocalDateTime.now();
// ✅ 正确:静态复用,线程安全
static final DateTimeFormatter LOG_FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSS");
String logTime = now.format(LOG_FORMATTER); // "2025-12-30 08:54:12.345"
LocalDateTime parsed = LocalDateTime.parse("2025-12-30 08:54:12.345", LOG_FORMATTER);
真正麻烦的从来不是写对一行 ofPattern,而是在跨系统、跨语言、跨时区场景下,确保格式器能稳定扛住各种输入变体——所以别省那几行 trim() 和 withLocale()。