# JDK 1.8 java.time 包

# 概述

  • java 1.8之前的日期类存在几个问题:

    • 线程不安全,java.util.Date 这个类线程不安全,是可变的。导致了需要二次封装。
    • 时间处理麻烦,默认时间从1900开始,不支持国际化,不提供时区支持。
    • 设计不好,比如java.utiljava.sql中的日期类,名称都是一样,容易导错包。
  • java 1.8以后的java.time包,日期类不可变且线程安全的。吸取了第三方jada-time

# time关键类

  • Instant 它代表的是时间戳,这里的时间戳默认的是0时区的,所以比北京时间少8个时区

  • LocalDate 不包含时间的日期

  • LocalTime 时间

  • LocalDateTime 日期和时间

  • ZonedDateTime 包含时区的完整的日期时间,偏移量是以UTC/格林威治时间为基准的,如:2018-10-08T18:12:38.547+08:00[Asia/Shanghai]。

# 常用API

# LocalDate、LocalTime、LocalDateTime

  • 这三个类分别表示日期、时间、日期时间,其API具有相似性。用LocalDateTime来举例它们的共性

# LocalDateTime

 //获取此刻的日期时间
LocalDateTime localDateTime = LocalDateTime.now();
/**
* 获取当前日期时间的明天、昨天
* 同理还有年、月 、日 时、分、秒的加减
*/
LocalDateTime tomorrow = localDateTime.plusDays(1L);
LocalDateTime yesterday = localDateTime.minusDays(1L);
/**
* 获取本月、本年的日期的API LocalDate LocalDateTime都有,LocalTime有类似的获取时间的API
*/
//第5天是本月的哪天
LocalDateTime withDayOfMonth = localDateTime.withDayOfMonth(5);
//第120天时本年的哪天
LocalDateTime withDayOfYear = localDateTime.withDayOfYear(120);

/**
* 计算两个日期时间的差值, (后面的 - 前面的 +2)
* 同理还有年、月、日、时、分、秒的差值比较
* LocalTime LocalDate都有此API,只是LocalTime不可计算年月日 ,LocalDate不可计算时分秒
*/
long untilDay = yesterday.until(tomorrow, ChronoUnit.DAYS);//2
/**
*日期的比较,返回一个boolean值,也有isAfter与此相反
* LocalTime LocalDate都有此API
*/
boolean before = yesterday.isBefore(tomorrow);//true
//转时间戳,需要设置东八区 +8
long timestamp = localDateTime.toInstant(ZoneOffset.ofHours(8)).toEpochMilli();
/**
* 用DateTimeFormatter,把LocalDateTime转成想要的日期格式字符串 (HH:mm:ss是24小时制)
* LocalDate 也适用
*/
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyyMMdd HH:mm:ss");
LocalDateTime time = LocalDateTime.now();
String format = dateTimeFormatter.format(time);
//把字符串转为LocalDate
LocalDate localDate = LocalDate.parse("20180813",DateTimeFormatter.ofPattern("yyyyMMdd"));
  • TemporalAdjuster该类是一个计算用的类,提供了各种各样的计算方法。比如某个月的第一天,某个月的最后一天,某一年的第一天,某一年的第几天等各种计算方法。同样用LocalDateTime举例
LocalDateTime localDateTime = LocalDateTime.now();
/**
* TemporalAdjusters类 适用LocalDate LocalDateTime
*/
//本月第一天
LocalDateTime firstDayOfMonth = localDateTime.with(TemporalAdjusters.firstDayOfMonth());
//本年最后一天
LocalDateTime lastDayOfMonth = localDateTime.with(TemporalAdjusters.lastDayOfYear());
//下月第一天
LocalDateTime firstDayOfNextMonth = localDateTime.with(TemporalAdjusters.firstDayOfNextMonth());
  • PeriodDuration Period是基于ISO-8601标准的日期系统,用于计算两个日期间的年,月,日的差值。比如'2年,3个月,4天';而Duration和Period很像,但Duration计算的是两个日期间的秒,纳秒的值,是一种更为精确的计算方式。
    • Period.between(LocalDate day1,LocalDate day2) 底层是 until方法实现,只是只接收LocalDate参数.
    • Duration.between(LocalTime time1,LocalTime time2)计算两个时间之间的秒数,纳秒数。有一点需要注意,Duration中必须要支持秒数,如果没有的话,将会报错.
  • ChronoUnit 实现功能类型Period和Duration,但如果我们看它的底层实现,就可以看到它基本是基于Duration来实现的.
LocalDateTime oldDate = LocalDateTime.of(2017, Month.AUGUST, 31, 10, 20, 55);
LocalDateTime newDate = LocalDateTime.of(2018, Month.NOVEMBER, 9, 10, 21, 56);

long years = ChronoUnit.YEARS.between(oldDate, newDate);
long months = ChronoUnit.MONTHS.between(oldDate, newDate);
long weeks = ChronoUnit.WEEKS.between(oldDate, newDate);
long days = ChronoUnit.DAYS.between(oldDate, newDate);
long hours = ChronoUnit.HOURS.between(oldDate, newDate);
long minutes = ChronoUnit.MINUTES.between(oldDate, newDate);
long seconds = ChronoUnit.SECONDS.between(oldDate, newDate);
long milis = ChronoUnit.MILLIS.between(oldDate, newDate);
long nano = ChronoUnit.NANOS.between(oldDate, newDate);

+Clock时钟系统,用于查找当前时刻。通过指定一个时区,我们可以获取到当前的时刻。

/**
* Clock 可获取系统或者自主选择时区,获取时区码,时间戳,时区本地的时间
*/
// 系统默认
Clock systemDefaultClock = Clock.systemDefaultZone();
// 世界协调时UTC
Clock systemUTCClock = Clock.systemUTC();
//芝加哥
Clock clock = Clock.system(ZoneId.of(ZoneId.SHORT_IDS.get("CST")));
LocalDateTime.now(clock);//2019-08-29T22:54:45.515
clock.millis();//1567137013253
clock.getZone();//America/Chicago

# Instant

/**
*获取0时区的时间戳
*打印输出 2019-08-29T09:45:28.164Z
*/
Instant now = Instant.now();

# ZonedDateTime

+ZonedDateTime 类似LocalDateTime,只是相比LocalDateTime,多了一个时区的概念

# java.time.Instantjava.util.Date 的转换

  • java 1.8 java.util.Date做了兼容,增加了两个方法Date.from(Instant instant)``toInstant()
    • Instant转为Date
public static Date from(Instant instant) {
    try {
        return new Date(instant.toEpochMilli());
    } catch (ArithmeticException ex) {
        throw new IllegalArgumentException(ex);
    }
}
  • Date转为Instant
public Instant toInstant() {
    return Instant.ofEpochMilli(getTime());
}

# java.time.LocalDateTimejava.util.Date 的转换

//LocalDateTime 转 Date
//转时间戳
long timestamp = localDateTime.toInstant(ZoneOffset.ofHours(8)).toEpochMilli();
//转Date
Date date = new Date(timestamp);

//  Date 转 LocalDateTime
//获取时间戳
Date date = new Date();
long time = date.getTime();
//转LocalDateTime
LocalDateTime localDateTime = Instant.ofEpochMilli(timestamp).atZone(ZoneOffset.ofHours(8)).toLocalDateTime();

# 总结

  • Instant.now()的时候注意时区问题。

  • time包下的localDate localTime localDateTime不需要类型转换都可以直接保存数据库,mysql 4.2版本以上。分别对应的类型是 date、time、 dateTime(timestamp)。

  • java.time包下其实还有一些其他的类与方法,如查询是否是闰年,日期比较,获取某一年的某一天等等。对日期时间的操作优先考虑time包下是否已有实现。