我知道:
- Instant 是用于计算的“技术”时间戳表示(纳秒)。
- LocalDateTime 是日期/时钟表示,包括人类的时区。
最后,IMO 都可以作为大多数应用程序用例的类型。举个例子:目前我正在运行一个批处理作业,我需要根据日期计算下一次运行,我正在努力寻找这两种类型之间的优缺点(除了 Instant 的纳秒精度优势和时区部分本地日期时间)。
您能否举出一些应用示例,其中只应使用 Instant 或 LocalDateTime?
编辑:小心误读 LocalDateTime 关于精度和时区的文档
原文由 manuel aldana 发布,翻译遵循 CC BY-SA 4.0 许可协议
tl;博士
Instant
和LocalDateTime
是两种完全不同的动物:一种代表片刻,另一种不代表。Instant
代表时刻,时间线中的特定点。LocalDateTime
代表日期和时间。但是缺少时区或与 UTC 的偏移量, 此类 不能 代表 moment 。它代表了大约 26 到 27 小时范围内的 潜在 时刻,即全球所有时区的范围。LocalDateTime
值 本质上是不明确 的。不正确的推定
您的说法不正确: A
LocalDateTime
has no time zone 。没有时区是该课程的重点。引用该类的文档:
所以
Local…
表示“未分区,无偏移”。Instant
Instant
是 UTC 时间线上的一个时刻,自 1970 年 UTC 第一个时刻的纪元以来的 纳秒 计数(基本上,请参阅类文档了解详细信息)。由于您的大部分业务逻辑、数据存储和数据交换都应该使用 UTC,因此这是一个经常使用的方便类。OffsetDateTime
类
OffsetDateTime
类将时刻表示为日期和时间,其上下文比 UTC 早或晚一些小时-分钟-秒。偏移量,即小时-分钟-秒数,由ZoneOffset
类表示。如果小时-分钟-秒数为零,则
OffsetDateTime
表示与Instant
相同的 UTC 时刻。ZoneOffset
ZoneOffset
类表示与 UTC 的 偏移量,比 UTC 早或晚于 UTC 数小时-分钟-秒。A
ZoneOffset
只是几个小时-分钟-秒,仅此而已。一个区域要多得多,它有一个名称和一个偏移变化的历史。因此,使用区域总是比仅使用偏移量更可取。ZoneId
时区 由
ZoneId
类表示。例如, 巴黎 的新一天比 蒙特利尔 更早。所以我们需要移动时钟的指针以更好地反映给定区域的 中午(当太阳直接在头顶时)。从西欧/非洲的 UTC 线向东/向西越远,偏移量越大。
时区是一组用于处理当地社区或地区实施的调整和异常情况的规则。最常见的异常是被称为 夏令时 (DST) 的非常流行的疯狂。
时区具有过去规则、现在规则和在不久的将来确认的规则的历史。
这些规则的变化比您预期的要频繁。确保保持你的日期时间库的规则,通常是 ‘tz’ 数据库 的副本,是最新的。随着 Oracle 发布了 Timezone Updater Tool ,在 Java 8 中保持最新比以往任何时候都容易。
Specify a proper time zone name in the format of
Continent/Region
, such asAmerica/Montreal
,Africa/Casablanca
, orPacific/Auckland
.切勿使用 2-4 个字母的缩写,例如EST
或IST
因为它们 不是 真正的时区,不是标准化的,甚至不是唯一的(!)。ZonedDateTime
将
ZonedDateTime
从概念上视为Instant
并分配有ZoneId
。要捕捉特定地区(时区)人们使用的挂钟时间中看到的当前时刻:
几乎所有的后端、数据库、业务逻辑、数据持久性、数据交换都应该使用 UTC。但是为了向用户展示,您需要调整到用户期望的时区。这是
ZonedDateTime
类和用于生成这些日期时间值的字符串表示形式的 格式化程序类 的目的。您可以使用
DateTimeFormatter
生成本地化格式的文本。LocalDate
,LocalTime
,LocalDateTime
“本地”日期时间类
LocalDateTime
、LocalDate
、LocalTime
是另一种生物。它们不依赖于任何一个地区或时区。它们与时间线无关。在您将它们应用到某个地方以在时间线上找到一个点之前, 它们没有真正的意义。这些类名中的“Local”这个词对于外行来说可能是违反直觉的。这个词的意思是 任何 地方,或 每一个 地方,但 不是 一个特定的地方。
因此,对于商业应用程序,“本地”类型并不经常使用,因为它们仅代表可能日期或时间的一般概念,而不是时间线上的特定时刻。商业应用程序往往关心发票到达的确切时间、运输的产品、雇用员工或出租车离开车库的确切时间。所以商业应用程序开发人员最常使用
Instant
和ZonedDateTime
类。那么我们什么时候使用
LocalDateTime
?在三种情况下:请注意,这三种情况都没有涉及时间线上的某个特定点,这些都不是片刻。
一天中的一个时间,多个时刻
有时我们想表示某个日期的某个时间,但希望将其应用于跨时区的多个地点。
例如,“圣诞节从 2015 年 12 月 25 日午夜开始”是
LocalDateTime
。巴黎的午夜罢工时间与蒙特利尔不同, 西雅图 和 奥克兰 也不同。另一个例子,“Acme 公司的政策是全球每个工厂的午餐时间从下午 12:30 开始”是
LocalTime
。要获得真正的意义,您需要将其应用于时间线,以计算 斯图加特 工厂的 12:30 或 拉巴特 工厂的 12:30 或 悉尼 工厂的 12:30 的时刻。预约
使用
LocalDateTime
的另一种情况是预订未来的活动(例如:牙医预约)。这些约会在未来可能已经足够遥远,以至于你冒着政治家重新定义时区的风险。政客们往往很少预先警告,甚至根本没有警告。如果你的意思是“明年 1 月 23 日下午 3 点”,不管政客们如何玩弄时钟,那么你就不能记录一个时刻——如果该地区采用或取消夏令时,那将看到下午 3 点变成下午 2 点或 4 点,例如。对于约会,存储一个
LocalDateTime
和一个ZoneId
,分开保存。稍后,在生成计划时,通过调用LocalDateTime::atZone( ZoneId )
即时确定时刻以生成ZonedDateTime
对象。如果需要,您可以调整为 UTC。从 —
Instant
ZonedDateTime
。未知区域
有些人可能会在时区或偏移量未知的情况下使用
LocalDateTime
。我认为这种情况是不恰当和不明智的。如果打算使用区域或偏移量但未确定,则您的数据不正确。这就像在不知道预期货币(美元、英镑、欧元等)的情况下存储产品价格。不是一个好主意。
所有日期时间类型
为了完整起见,这里是所有可能的日期时间类型的表,包括 Java 中的现代和传统类型,以及 SQL 标准定义的类型。这可能有助于将
Instant
和LocalDateTime
类放在更大的上下文中。请注意 Java 团队在设计 JDBC 4.2 时做出的奇怪选择。他们选择支持所有 java.time 时间……除了两个最常用的类:
Instant
&ZonedDateTime
。但不用担心。我们可以轻松地来回转换。
转换
Instant
。转换
ZonedDateTime
。关于 java.time
java.time 框架内置于 Java 8 及更高版本中。这些类取代了麻烦的旧 旧 日期时间类,例如
java.util.Date
、Calendar
和SimpleDateFormat
。要了解更多信息,请参阅 Oracle 教程。并在 Stack Overflow 上搜索许多示例和解释。规范是 JSR 310 。
现在处于 维护模式 的 Joda-Time 项目建议迁移到 java.time 类。
您可以直接与您的数据库交换 java.time 对象。使用符合 JDBC 4.2 或更高版本的 JDBC 驱动程序。不需要字符串,不需要
java.sql.*
类。 Hibernate 5 & JPA 2.2 支持 java.time 。从哪里获得 java.time 类?
ThreeTen-Extra 项目通过附加类扩展了 java.time。该项目是未来可能添加到 java.time 的试验场。您可以在这里找到一些有用的类,例如
Interval
、YearWeek
、YearQuarter
等等。