How to parse/format dates with LocalDateTime? (Java 8)
Java 8添加了一个新的java.time API来处理日期和时间(JSR 310)。
我有日期和时间作为字符串(例如
在我使用
解析日期和时间
要从字符串创建
1 2 3 | String str ="1986-04-08 12:30"; DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm"); LocalDateTime dateTime = LocalDateTime.parse(str, formatter); |
格式化日期和时间
要从
1 2 3 | DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm"); LocalDateTime dateTime = LocalDateTime.of(1986, Month.APRIL, 8, 12, 30); String formattedDateTime = dateTime.format(formatter); //"1986-04-08 12:30" |
请注意,在
如果
例如,
1 2 3 4 5 6 7 |
输出,
只有在必须处理其他日期模式时才使用
例如,dd MMM uuuu表示月份的日期(两位数),月份名称的三个字母(Jan,Feb,Mar,...)和四位数年份:
1 2 3 4 |
产量
1 | 04 Aug 2015 parses to 2015-08-04 |
还记得
1 2 3 4 |
产量
1 | 2015-08-04 formats as 04 Aug 2015 |
(请参阅格式化和解析DateFormatter的模式的完整列表)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 | Symbol Meaning Presentation Examples ------ ------- ------------ ------- G era text AD; Anno Domini; A u year year 2004; 04 y year-of-era year 2004; 04 D day-of-year number 189 M/L month-of-year number/text 7; 07; Jul; July; J d day-of-month number 10 Q/q quarter-of-year number/text 3; 03; Q3; 3rd quarter Y week-based-year year 1996; 96 w week-of-week-based-year number 27 W week-of-month number 4 E day-of-week text Tue; Tuesday; T e/c localized day-of-week number/text 2; 02; Tue; Tuesday; T F week-of-month number 3 a am-pm-of-day text PM h clock-hour-of-am-pm (1-12) number 12 K hour-of-am-pm (0-11) number 0 k clock-hour-of-am-pm (1-24) number 0 H hour-of-day (0-23) number 0 m minute-of-hour number 30 s second-of-minute number 55 S fraction-of-second fraction 978 A milli-of-day number 1234 n nano-of-second number 987654321 N nano-of-day number 1234000000 V time-zone ID zone-id America/Los_Angeles; Z; -08:30 z time-zone name zone-name Pacific Standard Time; PST O localized zone-offset offset-O GMT+8; GMT+08:00; UTC-08:00; X zone-offset 'Z' for zero offset-X Z; -08; -0830; -08:30; -083015; -08:30:15; x zone-offset offset-x +0000; -08; -0830; -08:30; -083015; -08:30:15; Z zone-offset offset-Z +0000; -0800; -08:00; p pad next pad modifier 1 ' escape for text delimiter '' single quote literal ' [ optional section start ] optional section end # reserved for future use { reserved for future use } reserved for future use |
上面的两个答案都很好地解释了关于字符串模式的问题但是,如果您正在使用ISO 8601,则无需应用
将LocalDateTime转换为时区ISO8601字符串
1 2 3 | LocalDateTime ldt = LocalDateTime.now(); ZonedDateTime zdt = ldt.atZone(ZoneOffset.UTC); //you might use a different zone String iso8601 = zdt.toString(); |
从ISO8601 String转换回LocalDateTime
1 2 3 | String iso8601 ="2016-02-14T18:32:04.150Z"; ZonedDateTime zdt = ZonedDateTime.parse(iso8601); LocalDateTime ldt = zdt.toLocalDateTime(); |
将具有日期和时间的字符串解析为特定时间点(Java将其称为"
然而,这种复杂性带来了很多困惑。
理解日期解析的关键是:
为什么Java有很多方法来解析日期
幸运的是,整个(商业)世界设法使用相同的。
......为什么
有时区。
时区基本上是地球表面的"条纹"* [1],其权限遵循相同的规则,它何时具有时间偏移。这包括夏令时规则。
各个地区的时区随时间而变化,主要取决于谁征服了谁。一个时区的规则也随着时间的推移而变化。
有时间抵消。这与时区不同,因为时区可以是例如时区。"布拉格",但有夏季时间抵消和冬季时间抵消。
如果您获得带有时区的时间戳,则偏移量可能会有所不同,具体取决于它所在年份的哪一部分。在闰日期间,时间戳可能意味着2个不同的时间,因此如果没有其他信息,则无法可靠地转换。
注意:按时间戳我的意思是"包含日期和/或时间的字符串,可选择带有时区和/或时间偏移量。"
多个时区可以在某些时段共享相同的时间偏移。例如,当夏令时偏移无效时,GMT / UTC时区与"伦敦"时区相同。
为了使它更复杂(但这对你的用例来说并不重要):
更新并不总是保留历史和未来时间戳的先前行为。因此,当在不同版本的软件上运行时,对比某些时区的两个时间戳进行解析可能会产生不同的结果。这也适用于比较受影响的时区和其他时区。
如果这会导致软件出现错误,请考虑使用一些没有复杂规则的时间戳,例如UNIX时间戳。
由于7,对于未来的日期,我们无法准确地转换日期。因此,例如,当前解析
JDK的API随着当代需求而发展
如何在Java的
确定要解析时间戳的类型
在使用时间戳字符串时,您需要知道它包含哪些信息。这是至关重要的一点。如果你没有做到这一点,你最终会得到一个神秘的例外,例如"无法创建即时"或"区域偏移丢失"或"未知区域ID"等。
它包含日期和时间吗?
它有时间偏移吗?
时间偏移是
对于这些时间戳,您使用
它有时区吗?
对于这些时间戳,您使用
区域由。指定
时区列表由ICAAN支持的"TZ数据库"编译。
根据
如果只有TZ的时间戳落入时间偏移变化的闰日,那么它是不明确的,并且解释是
如果两者都没有,则假设或忽略缺失的上下文。消费者必须做出决定。因此需要将其解析为
部分时间信息
如果您有完整的信息,可以获得
弄清楚如何解析它
有关
预先创建的
如果你有一些特殊的格式,那么你可以创建自己的DateTimeFormatter(它也是一个解析器)。
1 2 3 4 | private static final DateTimeFormatter TIMESTAMP_PARSER = new DateTimeFormatterBuilder() .parseCaseInsensitive() .append(DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SX")) .toFormatter(); |
我建议查看
TemporalAccessor
现在,经常出现的错误是进入
1 2 | // No need for this! TemporalAccessor ta = TIMESTAMP_PARSER.parse("2011-... etc"); |
但是,配备上一节的知识,您可以方便地解析您需要的类型:
1 | OffsetDateTime myTimestamp = OffsetDateTime.parse("2011-12-03T10:15:30.123457Z", TIMESTAMP_PARSER); |
你实际上并不需要
1 | OffsetDateTime myTimestamp = OffsetDateTime.parse("2011-12-03T10:15:30.123457Z"); |
关于
我希望我能理解你的灵魂:)
注意:Java 3和7有一个
[1]不仅仅是它们不是条纹,还有一些奇怪的极端。 例如,一些邻近的太平洋岛屿有+14:00和-11:00时区。 这意味着,虽然在一个岛上,有一个5月1日下午,在另一个岛上,到目前为止,仍然是4月30日下午12点(如果我算得正确:))
好。