Java高级编程--JDK8新增的日期时间API

在开发过程中我们常常会碰到要处理时间的需求,前篇介绍了JDK8之前的日期时间API的使用,但是这些API中有很多方法都已经过时了,不再适用或效率不高,难以适应现在的需求,本篇博客将介绍在JDK8中新增的日期时间API的使用。


JDK8之前的日期时间API↷传送门


目录:

☍ LocalDate、LocalTime、LocalDateTime类

☍ Instant瞬时类

☍ DateTimeFormat格式化与解析类

☍ 其他日期API


新日期时间API背景:

JDK 1.0中包含了一个java.util.Date类,但是它的大多数方法已经在JDK 1.1引入Calendar类之后被弃用了。而Calendar并不比Date好多少。它们面临的问题是:

☃ 可变性:对原对象的操作会导致对象发生变化,像日期和时间这样的类应该是不可变的,应该是操作后返回新的对象。

☃ 偏移性:Date中的年份是从1900开始的,而月份都从0开始。

☃ SimpleDateTime格式化只对Date有用,Calendar则不行。

☃ 此外,它们也不是线程安全的;不能处理闰秒等。

Java 8中引入的java.time API 已经纠正了
过去的缺陷,将来很长一段时间内它都会为我们服务。

新的 java.time 中包含了所有关于本地日期(LocalDate)、本地时间(LocalTime)、本地日期时间(LocalDateTime)、时区(ZonedDateTime)和持续时间(Duration)的类。


▾ LocalDate、LocalTime、LocalDateTime类

LocalDate、LocalTime、LocalDateTime 类是其中较重要的几个类,它们的实例
是不可变的对象,分别表示使用 ISO-8601日历系统的日期、时间、日期和时间。
它们提供了简单的本地日期或时间,不包含与时区
相关的信息。

➣ LocalDate代表IOS格式(yyyy-MM-dd)的日期:如:2020-06-16

➣ LocalTime表示一个时间,而不是日期,如:08:05:58.464

➣ LocalDateTime是用来表示日期和时间的,这是一个最常用的类之一,如:2020-06-16T08:05:58.464

注:ISO-8601日历系统是国际标准化组织制定的现代公民的日期和时间的表示
法,也就是公历

常用方法:

/**
 * 描述:JDK8中的日期相关API的使用
 *  1、LocalDateTime类相较于LocalDte和LocalTime使用的多一些  (具有类似String的不变性,不改变原对象,返回新的信息)
 *      LocalDateTime.now();   获取当前日期/时间的方法
 *      of(x,y...); 设置日期时间信息
 *      getXxx(); 获取指定信息内容,如年、月、日等
 *      withXxx(value):设置日期时间信息的某个属性,如设置日,设置小时等
 *      plusXxx(value):增加某个属性的值,负数为减  返回新的对象
 */
public void testLocalDateTimeAPI1(){
     /*now()方法,获取当前日期时间信息*/
     System.out.println("-----------now()方法---------");
     LocalDate localDate = LocalDate.now();   //获取当前日期
     LocalTime localTime = LocalTime.now();   //获取当前时间
     LocalDateTime localDateTime = LocalDateTime.now();   //获取当前日期+时间
     System.out.println("LocalDate:" + localDate);    //LocalDate:2020-06-16
     System.out.println("LocalTime:" + localTime);    //LocalTime:08:05:58.464
     System.out.println("localDateTime:" + localDateTime);  //localDateTime:2020-06-16T08:05:58.464

     /*of(x,y...)方法,设置日期时间信息  不变性*/
     System.out.println("-----------of()方法---------");
     LocalDateTime localDateTime1 = LocalDateTime.of(2016,9,01,12,20,18,256);
     System.out.println(localDateTime);  //2020-06-16T08:05:58.464
     System.out.println(localDateTime1);  //2016-09-01T12:20:18.000000256

     /*getXxx()方法,获取指定信息内容,如年月日等*/
     System.out.println("-----------getXxx()方法---------");
     int year = localDate.getYear();
     int month = localDate.getMonthValue();
     int day = localDateTime.getDayOfMonth();
     int hour = localTime.getHour();
     System.out.println("当前日期:" + year + "-" + month + "-" + day + " " + hour + ":"+localTime.getMinute()+":"+localTime.getSecond());
     //当前日期:2020-6-16 8:5:58
        
     /*withXxx(value):设置日期时间信息的某个属性,如设置日,设置小时等*/
     //体现LocalDate,LocalTime,LocalDateTime的不可变性
     System.out.println("-----------withXxx()方法---------");
     LocalDateTime localDateTime2 = localDateTime.withDayOfMonth(8);
     System.out.println(localDateTime1);   //2016-09-01T12:20:18.000000256
     System.out.println(localDateTime2);   //2020-06-08T08:05:58.464

     /*plusXxx(value):增加某个属性的值,负数为减  返回新的对象*/
     System.out.println("-----------plusXxx(value)方法---------");
     LocalDateTime localDateTime3 = localDateTime2.plusYears(1);
     System.out.println(localDateTime3);  //2021-06-08T08:05:58.464

     /*minusXxx(value):减少某个属性的值,负数为加 返回新的对象*/
     System.out.println("-----------minusXxx(value)方法---------");
     LocalDateTime localDateTime4 = localDateTime2.minusYears(1);
     System.out.println(localDateTime4);  //2019-06-08T08:05:58.464
}

▾ Instant瞬时类

☃ Instant:时间线上的一个瞬时点。 这可以被用来记录应用程序中的事件时间戳。

☃ Instant表示时间线上的一点,而不需要任何上下文信息,例如,时区。
概念上讲,它只是简单的表示自1970年1月1日0时0分0秒(UTC)开始的秒
数。

☃ java.time包是基于纳秒计算的,所以Instant的精度可以达到纳秒级。

✦ 1秒 = 1000毫秒 =10^6 微秒=10^9 纳秒

✦ 时间戳是指格林威治时间1970年01月01日 00时00分00秒( 北京时间1970年01月01日08时00分00秒)起至现在的总秒数。

public void testInstant(){
    Instant instant = Instant.now();  //获取本初子午线对应的标准时间
    System.out.println(instant);  //2020-06-13T11:16:44.956Z  和中国时区差将近8小时
    //2020-06-16T00:33:23.768Z
    
    //处理为中国时区:东8区
    OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours(8));
    System.out.println(offsetDateTime);   //2020-06-16T08:33:23.768+08:00
    //获取毫秒数  从1970-01-01 00:00:00开始
    Long millis = instant.toEpochMilli();
    System.out.println(millis);  //1592267603768
    System.out.println(new Date().getTime());  //1592267603898
    //将毫秒数转为Instant对象
    Instant instant1 = Instant.ofEpochMilli(1592267462055L);
    System.out.println(instant1.toEpochMilli());  //1592267462055
    //Date --> Instant
    Instant instant2 = new Date().toInstant();
    System.out.println(instant2);   //2020-06-16T00:33:23.898Z
    //Instant --> Date  通过毫秒数
    //LocalDateTime --> Instant
}

▾ DateTimeFormat格式化与解析类

DateTimeFormat类格式化方法:

➣ 预定义的标准格式。如:
ISO_LOCAL_DATE_TIME;ISO_LOCAL_DATE;ISO_LOCAL_TIM,与LocalDateTime、LocalDate、LocalTime类对应

➣ 本地化相关的格式。如:ofLocalizedDateTime(FormatStyle.LONG)

➣ 自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)

public void testDateTimeFormat(){
   //方式一:预定义的标准格式   ISO_LOCAL_DATE_TIME对应LocalDateTime类格式; ISO_LOCAL_DATE对应LocalDate类格式; ISO_LOCAL_TIME对应LocalTime类格式
   DateTimeFormatter dtf = DateTimeFormatter.ISO_LOCAL_DATE;
   //格式化:LocalDateTime/LocalDate/LocalTime --> 字符串
   LocalDateTime localDateTime = LocalDateTime.now();
   System.out.println("localDate:" + localDateTime);   //localDate:2020-06-16T08:43:54.724
   String timeStr1 = dtf.format(localDateTime);
   System.out.println("timeStr1:" + timeStr1);    //timeStr1:2020-06-16
   //解析: 逆过程  字符串 --> TemporalAccessor抽象类
   TemporalAccessor parse1 = dtf.parse(timeStr1);  
   System.out.println("parse1:"+parse1);  //parse1:{},ISO resolved to 2020-06-16

   //方式二:本地化相关的格式。
   //  如:ofLocalizedDateTime()
   //  参数:FormatStyle.LONG/FormatStyle.MEDIUM/FormatStyle.SHORT   适用于LocalDateTime
   //  如:ofLocalizedDate()
   //  参数:FormatStyleFULL/FormatStyle.LONG/FormatStyle.MEDIUM/FormatStyle.SHORT  适用于LocalDate
   DateTimeFormatter formatter1 = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.LONG);
   //格式化
   String timeStr2 = formatter1.format(localDateTime);
   System.out.println("timeStr2:"+timeStr2);   //timeStr2:2020年6月16日 上午08时43分54秒

   //方式三:自定义格式化格式
   DateTimeFormatter formatter2 = DateTimeFormatter.ofPattern("yyyy-MM--dd a HH:mm:ss");
   //格式化
   String timeStr3 = formatter2.format(LocalDateTime.now());
   System.out.println("timeStr3:" + timeStr3);   //timeStr3:2020-06--16 上午 08:43:54
   //解析
   TemporalAccessor parse2 = formatter2.parse(timeStr3);
   //TemporalAccessor --> LocalXxx
   LocalDateTime localDateTime1 = LocalDateTime.from(parse2);
   System.out.println("parse2:"+localDateTime1);  //parse2:2020-06-16T08:43:54

   //直接转化字符串,parse(CharSequence text) 或 parse(CharSequence text, DateTimeFormatter formatter)
   LocalDate localDate = LocalDate.parse("2020-06-13");
   System.out.println("localDate:"+localDate);  //localDate:2020-06-13
   LocalDate localDate1 = LocalDate.parse("2020-06-13",DateTimeFormatter.ofPattern("yyyy-MM-dd"));
   System.out.println("localDate1:" + localDate1);   //localDate1:2020-06-13
}

▾ 其他日期API

✦ 此部分了解即可,用的时候知道有这些API就行。

☄ ZoneId:该类中包含了所有的时区信息,一个时区的ID,如 Europe/Paris

☄ ZonedDateTime:一个在ISO-8601日历系统时区的日期时间,如 2007-12-03T10:15:30+01:00 Europe/Paris

   ➣ 其中每个时区都对应着ID,地区ID都为“{区域}/{城市}”的格式,例如:
Asia/Shanghai等

☄ Clock:使用时区提供对当前即时、日期和时间的访问的时钟

☄ 日期间隔:Period,用于计算两个“日期”间隔

☄ TemporalAdjuster : 时间校正器。有时我们可能需要获取例如:将日期调整到“下一个工作日”等操作。

 ☄ TemporalAdjusters : 该类通过静态方法(firstDayOfXxx()/lastDayOfXxx()/nextXxx())提供了大量的常用TemporalAdjuster 的实现。

public void otherTimeAPI(){
    //Duration:用于计算两个“时间”间隔,以秒和纳秒为基准
    System.out.println("-------Duration--------");
    LocalTime localTime = LocalTime.now();
    LocalTime localTime1 = LocalTime.of(8, 23, 32);
    System.out.println(localTime + "+++++" + localTime1);  //09:24:37.366+++++08:23:32
    //between():静态方法,返回Duration对象,表示两个时间的间隔
    Duration duration = Duration.between(localTime1, localTime);
    //时间间隔对象Duration
    System.out.println("duration:" + duration);   //duration:PT1H1M5.366S
    //将间隔对象转换为秒
    System.out.println("duration.getSeconds():" + duration.getSeconds());  				     //duration.getSeconds():3665
    //获取秒数小数点后的纳秒级数
    System.out.println("duration.getNano():" + duration.getNano());
    LocalDateTime localDateTime = LocalDateTime.of(2016, 6, 12, 15, 23, 32);
    LocalDateTime localDateTime1 = LocalDateTime.of(2017, 6, 12, 15, 23, 32);
    Duration duration1 = Duration.between(localDateTime, localDateTime1);
    //time1-time2间隔天数
    System.out.println("duration1.toDays():" + duration1.toDays()); //duration1.toDays():365
    System.out.println();

    //Period:用于计算两个“日期”间隔,以年、月、日衡量
    System.out.println("-------Period--------");
    LocalDate localDate = LocalDate.now();
    LocalDate localDate1 = LocalDate.of(2028, 3, 18);
    //period日期间隔类
    Period period = Period.between(localDate, localDate1);
    System.out.println("period:" + period);
    //间隔年份
    System.out.println("period.getYears():" + period.getYears());
    //间隔月份
    System.out.println("period.getMonths():" + period.getMonths());
    //间隔日
    System.out.println("period.getDays():" + period.getDays());
    //使用withXxx()设置period对象
    Period period1 = period.withYears(2);
    System.out.println("period1:" + period1);
    System.out.println();

    //ZonedDateTime:带时区的日期时间
    // ZonedDateTime的now():获取本时区的ZonedDateTime对象
    System.out.println("------ZonedDateTime-------");
    ZonedDateTime zonedDateTime = ZonedDateTime.now();
    System.out.println(zonedDateTime);
    // ZonedDateTime的now(ZoneId id):获取指定时区的ZonedDateTime对象
    ZonedDateTime zonedDateTime1 = ZonedDateTime.now(ZoneId.of("Asia/Tokyo"));
    System.out.println(zonedDateTime1);
    System.out.println();

    // TemporalAdjuster:时间校正器
    // 获取当前日期的下一个周日是哪天?
    System.out.println("--------TemporalAdjuster----------");
    TemporalAdjuster temporalAdjuster = TemporalAdjusters.next(DayOfWeek.SUNDAY);
    LocalDateTime localDateTime3 = LocalDateTime.now().with(temporalAdjuster);
    System.out.println("下一个周日是:"+localDateTime3);
    // 获取下一个工作日是哪天?
    LocalDate localDate4 = LocalDate.now().with(new TemporalAdjuster() {
        @Override
        public Temporal adjustInto(Temporal temporal) {
            LocalDate date = (LocalDate) temporal;
            if (date.getDayOfWeek().equals(DayOfWeek.FRIDAY)) {
                 return date.plusDays(3);
             } else if (date.getDayOfWeek().equals(DayOfWeek.SATURDAY)) {
                return date.plusDays(2);
             } else {
                return date.plusDays(1);
            }
        }
    });
    System.out.println("下一个工作日是:" + localDate4);
    System.out.println();

    //ZoneId:类中包含了所有的时区信息
    // ZoneId的getAvailableZoneIds():获取所有的ZoneId
    System.out.println("-------ZoneId--------");
    Set<String> zoneIds = ZoneId.getAvailableZoneIds();
    for (String s : zoneIds) {
        System.out.println(s);
    }
    // ZoneId的of():获取指定时区的时间
    LocalDateTime localDateTime2 = LocalDateTime.now(ZoneId.of("Asia/Tokyo"));
    System.out.println(localDateTime2);
}

本博客与CSDN博客༺ཌ༈君☠纤༈ད༻同步发布

原文地址:https://www.cnblogs.com/asio/p/13138963.html