
在java开发中,处理日期和时间字符串的转换是一项常见任务,但往往伴随着各种挑战。不同的系统和数据源可能提供多种格式的时间戳,例如包含时区偏移的rfc 1123格式("thu, 3 nov 2022 06:00:00 +0100")或更简洁的本地日期时间格式("01.11.2022 20:00:00")。将这些多样化的输入统一转换为特定的输出格式(如"03.11.2022")需要精确的解析和格式化策略。
传统的java.util.Date和java.text.SimpleDateFormat API在处理这些场景时存在诸多问题,例如线程不安全、API设计不直观以及时区处理复杂等。Java 8引入的java.time API(JSR-310)提供了现代、简洁且功能强大的解决方案,极大地简化了日期时间操作。本教程将专注于使用java.time API来解决多格式日期时间字符串的转换问题。
java.time API的核心在于其不可变的值对象和清晰的类型系统,如LocalDate(日期)、LocalTime(时间)、LocalDateTime(日期时间)、OffsetDateTime(带偏移量的日期时间)和ZonedDateTime(带时区的日期时间)。对于字符串的解析和格式化,DateTimeFormatter是关键工具。
DateTimeFormatter通过模式字符串来定义日期时间的格式。理解这些模式符号至关重要:
| 符号 | 含义 | 示例(英文Locale) | 备注 |
|---|---|---|---|
| d | 月中的日 | 1 到 31 | 单数字不补零,双数字补零。 |
| dd | 月中的日(两位) | 01 到 31 | 始终两位数,不足补零。 |
| M | 年中的月 | 1 到 12 | 单数字不补零,双数字补零。 |
| MM | 年中的月(两位) | 01 到 12 | 始终两位数,不足补零。 |
| MMM | 月份缩写 | Jan, Feb, Nov | 需要指定Locale。 |
| MMMM | 月份全称 | January, February, November | 需要指定Locale。 |
| u | 年份 | 2022 | java.time推荐使用,表示年-of-era。 |
| uu | 年份(两位) | 22 | |
| uuuu | 年份(四位) | 2022 | |
| H | 日中的小时(0-23) | 0 到 23 | 单数字不补零。 |
| HH | 日中的小时(0-23,两位) | 00 到 23 | 始终两位数,不足补零。 |
| h | AM/PM中的小时(1-12) | 1 到 12 | |
| mm | 分钟 | 00 到 59 | |
| ss | 秒 | 00 到 59 | |
| x | 偏移量 | +01, +0100, +01:00 | 用于解析带时区偏移的字符串。 |
| EEE | 星期几缩写 | Mon, Tue, Thu | 需要指定Locale。 |
| EEEE | 星期几全称 | Monday, Tuesday, Thursday | 需要指定Locale。 |
常见错误提示:
立即学习“Java免费学习笔记(深入)”;
当输入和输出的日期时间格式不同时,你需要为解析(parse)和格式化(format)操作分别创建DateTimeFormatter实例。解析器负责理解输入的字符串结构,而格式化器则负责将日期时间对象转换为目标字符串结构。
如果日期时间字符串包含月份或星期的名称(如Nov、Thu),则在创建DateTimeFormatter时必须指定Locale,以便正确识别这些文本信息。例如,Locale.ENGLISH适用于英文月份和星期缩写。
本节将通过具体的代码示例,演示如何将两种不同格式的输入字符串转换为目标格式"dd.MM.uuuu"。
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.OffsetDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Locale;
public class DateTimeConverter {
public static void main(String[] args) {
// 示例输入字符串
String firstInput = "Thu, 3 Nov 2022 06:00:00 +0100";
String secondInput = "01.11.2022 20:00:00";
String thirdInput = "9.28.2022 6:30:00"; // 非标准格式示例
// 1. 定义目标输出格式的DateTimeFormatter
DateTimeFormatter outputFormatter = DateTimeFormatter.ofPattern("dd.MM.uuuu");
System.out.println("--- 转换示例 ---");
// 案例一:RFC 1123格式的解析与转换
// 输入格式: "EEE, d MMM uuuu HH:mm:ss x"
// 包含英文月份和星期缩写,需要指定Locale.ENGLISH
DateTimeFormatter rfc1123Parser = DateTimeFormatter.ofPattern(
"EEE, d MMM uuuu HH:mm:ss x",
Locale.ENGLISH
);
// 或者使用内置的RFC 1123格式化器
// DateTimeFormatter rfc1123Parser = DateTimeFormatter.RFC_1123_DATE_TIME;
try {
// 解析为OffsetDateTime,因为它包含时区偏移信息
OffsetDateTime odt = OffsetDateTime.parse(firstInput, rfc1123Parser);
// 提取日期部分并格式化
String formattedDate = odt.format(outputFormatter);
System.out.println(firstInput + " ---> " + formattedDate);
} catch (java.time.format.DateTimeParseException e) {
System.err.println("解析 '" + firstInput + "' 失败: " + e.getMessage());
}
// 案例二:标准日期时间格式的解析与转换
// 输入格式: "dd.MM.uuuu HH:mm:ss"
DateTimeFormatter standardParser = DateTimeFormatter.ofPattern("dd.MM.uuuu HH:mm:ss");
try {
// 解析为LocalDateTime,因为它不包含时区偏移信息
LocalDateTime ldt = LocalDateTime.parse(secondInput, standardParser);
// 提取日期部分并格式化
String formattedDate = ldt.format(outputFormatter);
System.out.println(secondInput + " ---> " + formattedDate);
} catch (java.time.format.DateTimeParseException e) {
System.err.println("解析 '" + secondInput + "' 失败: " + e.getMessage());
}
// 案例三:处理非标准或模糊日期格式
// 输入格式: "9.28.2022 6:30:00"
// 注意:这里的"9.28"是月和日,而不是日和月。
// 且单数字的月、日、小时不补零,所以模式应为 "M.d.uuuu H:mm:ss"
DateTimeFormatter ambiguousParser = DateTimeFormatter.ofPattern("M.d.uuuu H:mm:ss");
try {
// 解析为LocalDateTime
LocalDateTime ldtThird = LocalDateTime.parse(thirdInput, ambiguousParser);
// 提取日期部分并格式化
String formattedDate = ldtThird.format(outputFormatter);
System.out.println(thirdInput + " ---> " + formattedDate);
} catch (java.time.format.DateTimeParseException e) {
System.err.println("解析 '" + thirdInput + "' 失败: " + e.getMessage());
}
}
}运行结果:
--- 转换示例 --- Thu, 3 Nov 2022 06:00:00 +0100 ---> 03.11.2022 01.11.2022 20:00:00 ---> 01.11.2022 9.28.2022 6:30:00 ---> 28.09.2022
代码解析:
通过java.time API及其核心类DateTimeFormatter,Java开发者可以高效、准确地处理各种日期时间字符串的解析与格式化需求。理解不同的日期时间类型、掌握模式符号的精确含义,并针对性地创建解析器和格式化器,是成功实现复杂日期时间转换的关键。遵循本教程中的最佳实践,将有助于构建更加健壮和可维护的日期时间处理逻辑。
以上就是Java日期时间字符串转换实战:从多格式到“dd.MM.uuuu”的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号