Java中的时间
你如果以为,Java中谈到时间仅仅就意味着java.util.Date那就大错特错了,Java中的时间其实可以说五花八门,Java8发布后又增加了一些新的用来表示日期和时间的类,那么我们在构建应用程序的时候到底应该用哪个类来呢?彼此之间又有什么区别?
1 关于时间的表示
通常由于文化和地区的不同,世界上各个地方的人们对于时间的表达方式都不尽相同,比如在中国以前用农历和十二时辰来表示,而在西方是二十四小时制(巧合的是正好一个时辰能对应上两个小时,据称与”高合成数“有关)。那么在计算机领域,在表示时间的时候也有不同的表示方法,比较常见的有:
- 通过计算当前时间到Jan 01 1970(Unix
Epoch)这一天(准确的说是00:00)经过的秒数来表示,例如
1573090869。 - 比较直接的方法就是直接以时分秒的形式表示当地时间,同时将当地的时区加进去,例如
2001-07-04T12:08:56.235-07:00
2 Java中的时间类
抛开时间戳不谈,在Java中专门用来表示时间的其他类有:
| Class | Since | Description |
|---|---|---|
| java.util.Date | JDK1.0 | 日期+时间 |
| java.util.Calendar | JDK1.1 | |
| java.sql.Date | 只包含日期 | |
| java.sql.Time | 只包含时间 | |
| java.sql.Timestamp | ||
| java.time.Instant | JDK1.8 | |
| java.time.LocalTime | JDK1.8 | |
| java.time.LocalDate | JDK1.8 | |
| java.time.OffsetTime | JDK1.8 | |
| java.time.OffsetDateTime | JDK1.8 | |
| java.time.ZonedDateTime | JDK1.8 |
在Java8之前,我们通常用java.util.Date来表示时间,虽然没啥需求实现不了的,但有着以下的问题:
- 时间类不统一,
java.util和java.sql包中都有关于时间的类,而时间格式化的又在java.text包中,有点乱的很 - 所有时间的类都是mutable的,非线程安全
所以Java8开始对时间进行了修改,使用起来将更加方便。通常来讲,对于需要处理时区问题的系统,OffsetDateTime是一个较好的选择,即包含了时间信息,又包含了时区的信息,可以得到准确的时间表述。官方文档中也建议使用:
It is intended that ZonedDateTime or Instant is used to model data in simpler applications. This class may be used when modeling date-time concepts in more detail, or when communicating to a database or in a network protocol.
在处理类似时间转换的时候,可以借助ZonedDateTime来实现。例如有一个yyyyMMddHHmmss格式的时间,但该时间是CST时间,这个时间有如下的特点:
- 在夏季时间相当于utc+5
- 在冬季相当于utc+6
而每年会有一个时间点去切换这个夏季时间和冬季时间(称之为day-saving),而且不是一个固定的日期(大致相当于按照第几个星期几来算的),要想把这个时间转换为标准的时间描述,可以采取这样的方式:
LocalDateTime localDateTime = LocalDateTime.parse(
"20191010095425", DateTimeFormatter.ofPattern("yyyyMMddHHmmss"));
ZonedDateTime zonedDateTime = ZonedDateTime.of(
localDateTime, TimeZone.getTimeZone("CST"));
OffsetDateTime result = zonedDateTime
.toOffsetDateTime()
.withOffsetSameInstant(ZoneOffset.UTC);