在阅读本文之前,请先看上一篇文章 有关机器时间、UTC时间、本地时间的总结。
Java新时间API中有三个特别重要的类,分别是Instant、LocalDateTime、ZonedDateTime,它们分别对应到上一篇文章中讲到的时间概念为:机器时间、无时区的本地时间、有时区的本地时间。
机器时间其实可以理解为UTC时间的另一种表现形式,其可以唯一确定时间线上的某一时刻。
无时区的本地时间因为没有时区信息,所以其无法唯一确定时间线上的某一时刻。
有时区的本地时间其实是在UTC时间的基础上加一些时间偏移,所以也是可以唯一确定时间线上的某一时刻。
Java的新时间API其实都是围绕这三个类来实现的,所以,彻底理解这三个类的目的及使用场景对于灵活使用Java新时间API来说非常重要。
下面我们用示例来讲解下Java的新时间API究竟怎么用。
- 获取当前时间的年月日等信息。
static void t1() {
LocalDateTime ldt = LocalDateTime.now();
System.out.println(ldt.getYear());
ZonedDateTime zdt = ZonedDateTime.now();
System.out.println(zdt.getYear());
// 对于获取当前时刻的human time信息(年月日时分秒)来说
// 用LocalDateTime或者ZonedDateTime都是一样的
// 他们的底层都是先获取machine time,然后再按照所在时区
// 将machine time转成human time
// ZonedDateTime比LocalDateTime的唯一区别就是携带了时区信息
// 但如果只是为了获取年月日等信息,时区是没用的
}
- 获取某一机器时间的年月日等信息。
static void t2() {
long millis = System.currentTimeMillis();
Instant instant = Instant.ofEpochMilli(millis);
LocalDateTime ldt = LocalDateTime.ofInstant(instant, ZoneId.systemDefault());
System.out.println(ldt.getYear());
ZonedDateTime zdt = ZonedDateTime.ofInstant(instant, ZoneId.systemDefault());
System.out.println(zdt.getYear());
// 机器时间要先转成Instant实例后,才能被Java新时间API使用
}
- 人类时间转机器时间。
static void t3() {
// 由于LocalDateTime没有时区信息,所以它无法直接转成机器时间
// 只能通过主动提供时区信息的方式才可以
LocalDateTime ldt = LocalDateTime.now();
long millis = ldt.toInstant(ZoneOffset.of("+08:00")).toEpochMilli();
System.out.println(millis);
// 由于ZonedDateTime内已经有时区信息了,所以它可以直接转成机器时间
ZonedDateTime zdt = ZonedDateTime.now();
millis = zdt.toInstant().toEpochMilli();
System.out.println(millis);
// 还需要注意的是,代表人类时间的类并不是直接转成的机器时间,而是通过Instant类间接完成
}
- 解析时间字符串。
static void t4() {
// 被解析的字符串不能有时区信息
LocalDateTime ldt = LocalDateTime.parse("2019-09-25T16:32:42");
System.out.println(ldt);
// 被解析的字符串必须有时区信息
ZonedDateTime zdt = ZonedDateTime.parse("2019-09-25T16:32:42+08:00");
System.out.println(zdt);
}
- 时间的运算。
static void t5() {
// LocalDateTime的时间加减就是纯粹的加减
LocalDateTime ldt = LocalDateTime.parse("2019-03-10T01:59:59");
System.out.println(ldt.plusHours(1).getHour()); // 输出:2
// ZonedDateTime的时间加减还会考虑时区信息
// 比如2019-03-10T02:00:00开始,美国开始施行 daylight saving time (夏令时)
// 他们的本地时间会向后拨一个小时,即:凌晨2点会变成凌晨3点
// 有关 daylight saving time 更多信息,请看以下文章:
// https://en.wikipedia.org/wiki/Daylight_saving_time
ZonedDateTime zdt = ldt.atZone(ZoneId.of("America/New_York"));
System.out.println(zdt.plusHours(1).getHour()); // 输出:3
// 所以说,如果涉及到时间的运算,要用ZonedDateTime
}
- 获取某一机器时间所属那天的零点的机器时间。
static long startOfDay(long millis) {
return LocalDate.ofInstant(Instant.ofEpochMilli(millis), ZoneId.systemDefault())
.atStartOfDay(ZoneId.systemDefault())
.toInstant()
.toEpochMilli();
}
- 获取某一机器时间所属那个星期的星期一零点的机器时间。
static long startOfWeek(long millis) {
return LocalDate.ofInstant(Instant.ofEpochMilli(millis), ZoneId.systemDefault())
.atStartOfDay(ZoneId.systemDefault())
.with(DayOfWeek.MONDAY)
.toInstant()
.toEpochMilli();
}
- 获取某一机器时间所属那个月的一号零点的机器时间。
static long startOfMonth(long millis) {
return LocalDate.ofInstant(Instant.ofEpochMilli(millis), ZoneId.systemDefault())
.atStartOfDay(ZoneId.systemDefault())
.withDayOfMonth(1)
.toInstant()
.toEpochMilli();
}
好,例子就这么多吧,我觉得到这里大家都差不多懂了。
有关Java新时间API更多介绍,请参考Java官方教程:
https://docs.oracle.com/javas...
希望对大家有所帮助。
完。
更多原创文章,请关注我微信公众号: