不幸的是,接受的答案具有误导性。事实上,
2012-06-01T00:00:00.000+02:00 = 2012-05-31T22:00:00Z
Z
右侧是零时区偏移的时区指示符。它代表 Zulu 并指定Etc/UTC
时区(时区偏移量为+00:00
小时)。
写2012-06-01T00:00:00.000+02:00
as 2012-06-01
,虽然只是一个函数调用,但对于任何依赖于时区的业务逻辑都是危险的,因为它可能在时区中具有不同的日期,具有不同的偏移值,例如如上所示。2012-06-01
只是一个LocalDate
应该用于跟踪诸如出生日期、婚礼日期等事件的文件。
java.time
旧的日期时间 API(java.util
日期时间类型及其格式类型SimpleDateFormat
等)已过时且容易出错。建议完全停止使用它并切换到java.time
现代日期时间 API *。
此外,下面引用的是Joda-Time 主页上的通知:
请注意,从 Java SE 8 开始,用户被要求迁移到 java.time (JSR-310) - JDK 的核心部分,它取代了这个项目。
使用java.time
现代 API 的解决方案:
如何解析给定的日期时间字符串:
给定的日期时间字符串有一个时区偏移量,因此它应该被解析为OffsetDateTime
. 由于现代日期时间 API 基于ISO 8601DateTimeFormatter
,并且只要日期时间字符串符合 ISO 8601 标准,就不需要显式使用对象。
OffsetDateTime odt = OffsetDateTime.parse("2012-06-01T00:00:00.000+02:00"); // 2012-06-01T00:00+02:00
如何在 UTC 中获取日期时间:
有多种方法。最简单的方法是将其转换为Instant
UTC 时间轴上的瞬时点。
Instant instant = odt.toInstant(); // 2012-05-31T22:00:00Z
或者,
OffsetDateTime odtUtc = odt.withOffsetSameInstant(ZoneOffset.UTC); // 2012-05-31T22:00Z
如何java.util.Date
摆脱它:
如果您需要java.util.Date
来自 的实例的实例OffsetDateTime
,您可以使用Date#from(Instant instant)
.
Date date = Date.from(instant); // Thu May 31 23:00:00 BST 2012 <--In my timezone
请注意,java.util.Date
对象不是像现代日期时间类型那样的真实日期时间对象;相反,它表示自称为“纪元” January 1, 1970, 00:00:00 GMT
(或 UTC)的标准基准时间以来的毫秒数。当您打印 的对象时java.util.Date
,其toString
方法会返回 JVM 时区中的日期时间,根据此毫秒值计算得出。如果您需要在不同的时区打印日期时间,则需要将时区设置为SimpleDateFormat
并从中获取格式化的字符串,例如
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSXXX", Locale.ENGLISH);
sdf.setTimeZone(TimeZone.getTimeZone("Etc/UTC"));
System.out.println(sdf.format(date)); // 2012-05-31T22:00:00.000Z
如何从中获取日期部分:
正如我已经解释过的,任何依赖于时区的业务逻辑都是危险的。但是,这只是一个简单的函数调用问题。
LocalDate localDate = odt.toLocalDate(); // 2012-06-01
从Trail: Date Time了解更多关于java.time
现代日期时间 API *的信息。
* 出于任何原因,如果您必须坚持使用 Java 6 或 Java 7,则可以使用ThreeTen-Backport,它将大部分java.time功能向后移植到 Java 6 和 7。如果您正在为 Android 项目和 Android API 工作level 仍然不符合 Java-8,请检查Java 8+ APIs available through desugaring和How to use ThreeTenABP in Android Project。