Java8 - 新的日期/时间 API

在 Java8 中,引入了一个新的日期时间 API 来弥补旧日期时间 API 的以下缺点。

  • 线程不安全 − java.util.Date 线程不是安全的,因此开发人员在使用日期时必须处理并发问题。 新的日期时间 API 是不可变的,并且没有 setter 方法。

  • 设计不佳 − 默认日期从1900开始,月从1开始,日从0开始,所以没有统一性。 旧的 API 没有那么直接的日期操作方法。 新的 API 为此类操作提供了许多实用方法。

  • 困难的时区处理 − 开发人员不得不编写大量代码来处理时区问题。 开发新的 API 时牢记特定领域的设计。

Java8 在 java.time 包下引入了一个新的日期时间 API。 以下是 java.time 包中引入的一些重要类。

  • Local − 简化的日期时间 API,没有复杂的时区处理。

  • Zoned − 用于处理各种时区的专用日期时间 API。


本地日期时间 API

LocalDate/LocalTime 和 LocalDateTime 类简化了不需要时区的开发。 让我们看看它们的实际效果。

使用您选择的任何编辑器(例如 C:\> JAVA)创建以下 java 程序。

Java8Tester.java

import java.time.LocalDate;
import java.time.LocalTime;
import java.time.LocalDateTime;
import java.time.Month;

public class Java8Tester {

   public static void main(String args[]) {
      Java8Tester java8tester = new Java8Tester();
      java8tester.testLocalDateTime();
   }
	
   public void testLocalDateTime() {
      // 获取当前日期和时间
      LocalDateTime currentTime = LocalDateTime.now();
      System.out.println("Current DateTime: " + currentTime);
		
      LocalDate date1 = currentTime.toLocalDate();
      System.out.println("date1: " + date1);
		
      Month month = currentTime.getMonth();
      int day = currentTime.getDayOfMonth();
      int seconds = currentTime.getSecond();
		
      System.out.println("Month: " + month +"day: " + day +"seconds: " + seconds);
		
      LocalDateTime date2 = currentTime.withDayOfMonth(10).withYear(2012);
      System.out.println("date2: " + date2);
		
      //2014 年 12 月 12 日
      LocalDate date3 = LocalDate.of(2014, Month.DECEMBER, 12);
      System.out.println("date3: " + date3);
		
      //22小时15分钟
      LocalTime date4 = LocalTime.of(22, 15);
      System.out.println("date4: " + date4);
		
      //解析一个字符串
      LocalTime date5 = LocalTime.parse("20:15:30");
      System.out.println("date5: " + date5);
   }
}

验证结果

使用javac编译器编译类,,如下所示 −

C:\JAVA>javac Java8Tester.java

现在按如下方式运行 Java8Tester −

C:\JAVA>java Java8Tester

它应该产生以下输出 −

Current DateTime: 2014-12-09T11:00:45.457
date1: 2014-12-09
Month: DECEMBERday: 9seconds: 45
date2: 2012-12-10T11:00:45.457
date3: 2014-12-12
date4: 22:15
date5: 20:15:30

分时区日期时间 API

当考虑时区时,将使用 Zoned date-time API。 让我们看看他们的行动。

使用您选择的任何编辑器(例如 C:\> JAVA)创建以下 Java 程序。

Java8Tester.java

import java.time.ZonedDateTime;
import java.time.ZoneId;

public class Java8Tester {

   public static void main(String args[]) {
      Java8Tester java8tester = new Java8Tester();
      java8tester.testZonedDateTime();
   }
	
   public void testZonedDateTime() {
      // 获取当前日期和时间
      ZonedDateTime date1 = ZonedDateTime.parse("2007-12-03T10:15:30+05:30[Asia/Karachi]");
      System.out.println("date1: " + date1);
		
      ZoneId id = ZoneId.of("Europe/Paris");
      System.out.println("ZoneId: " + id);
		
      ZoneId currentZone = ZoneId.systemDefault();
      System.out.println("CurrentZone: " + currentZone);
   }
}

验证结果

使用javac编译器编译类,,如下所示 −

C:\JAVA>javac Java8Tester.java

现在按如下方式运行 Java8Tester −

C:\JAVA>java Java8Tester

它应该产生以下输出 −

date1: 2007-12-03T10:15:30+05:00[Asia/Karachi]
ZoneId: Europe/Paris
CurrentZone: Etc/UTC

时间单元枚举

java.time.temporal.ChronoUnit 枚举是在 Java8 中添加的,以取代旧 API 中使用的整数值来表示日、月等。让我们看看它们的实际应用。

使用您选择的任何编辑器(例如 C:\> JAVA)创建以下 Java 程序。

Java8Tester.java

import java.time.LocalDate;
import java.time.temporal.ChronoUnit;

public class Java8Tester {

   public static void main(String args[]) {
      Java8Tester java8tester = new Java8Tester();
      java8tester.testChromoUnits();
   }
	
   public void testChromoUnits() {
      //获取当前日期
      LocalDate today = LocalDate.now();
      System.out.println("Current date: " + today);
		
      //在当前日期上加 1 周
      LocalDate nextWeek = today.plus(1, ChronoUnit.WEEKS);
      System.out.println("Next week: " + nextWeek);
		
      //在当前日期上加1个月
      LocalDate nextMonth = today.plus(1, ChronoUnit.MONTHS);
      System.out.println("Next month: " + nextMonth);
		
      //在当前日期上加1年
      LocalDate nextYear = today.plus(1, ChronoUnit.YEARS);
      System.out.println("Next year: " + nextYear);
		
      //在当前日期基础上增加 10 年
      LocalDate nextDecade = today.plus(1, ChronoUnit.DECADES);
      System.out.println("Date after ten year: " + nextDecade);
   }
}

验证结果

使用javac编译器编译类,,如下所示 −

C:\JAVA>javac Java8Tester.java

现在按如下方式运行 Java8Tester −

C:\JAVA>java Java8Tester

它应该产生以下结果 −

Current date: 2014-12-10
Next week: 2014-12-17
Next month: 2015-01-10
Next year: 2015-12-10
Date after ten year: 2024-12-10

期间和持续时间

在 Java8 中,引入了两个专门的类来处理时间差异。

  • Period − 它处理基于日期的时间量。

  • Duration − 它处理基于时间的时间量。

让我们看看他们的行动。

使用您选择的任何编辑器(例如 C:\> JAVA)创建以下 Java 程序。

Java8Tester.java

import java.time.temporal.ChronoUnit;

import java.time.LocalDate;
import java.time.LocalTime;
import java.time.Duration;
import java.time.Period;

public class Java8Tester {

   public static void main(String args[]) {
      Java8Tester java8tester = new Java8Tester();
      java8tester.testPeriod();
      java8tester.testDuration();
   }
	
   public void testPeriod() {
      //获取当前日期
      LocalDate date1 = LocalDate.now();
      System.out.println("Current date: " + date1);
		
      //在当前日期上加1个月
      LocalDate date2 = date1.plus(1, ChronoUnit.MONTHS);
      System.out.println("Next month: " + date2);
      
      Period period = Period.between(date2, date1);
      System.out.println("Period: " + period);
   }
	
   public void testDuration() {
      LocalTime time1 = LocalTime.now();
      Duration twoHours = Duration.ofHours(2);
		
      LocalTime time2 = time1.plus(twoHours);
      Duration duration = Duration.between(time1, time2);
		
      System.out.println("Duration: " + duration);
   }
}

验证结果

使用javac编译器编译类,,如下所示 −

C:\JAVA>javac Java8Tester.java

现在按如下方式运行 Java8Tester −

C:\JAVA>java Java8Tester

它应该产生以下输出 −

Current date: 2014-12-10
Next month: 2015-01-10
Period: P-1M
Duration: PT2H

时间调节器

TemporalAdjuster 用于执行日期计算。 例如,获取"每月的第二个星期六"或"下个星期二"。 让我们看看它们的实际效果。

使用您选择的任何编辑器(例如 C:\> JAVA)创建以下 Java 程序。

Java8Tester.java

import java.time.LocalDate;
import java.time.temporal.TemporalAdjusters;
import java.time.DayOfWeek;

public class Java8Tester {

   public static void main(String args[]) {
      Java8Tester java8tester = new Java8Tester();
      java8tester.testAdjusters();
   }
	
   public void testAdjusters() {
      //获取当前日期
      LocalDate date1 = LocalDate.now();
      System.out.println("Current date: " + date1);
		
      //获取下周二
      LocalDate nextTuesday = date1.with(TemporalAdjusters.next(DayOfWeek.TUESDAY));
      System.out.println("Next Tuesday on : " + nextTuesday);
		
      //获取下个月的第二个星期六
      LocalDate firstInYear = LocalDate.of(date1.getYear(),date1.getMonth(), 1);
      LocalDate secondSaturday = firstInYear.with(TemporalAdjusters.nextOrSame(
         DayOfWeek.SATURDAY)).with(TemporalAdjusters.next(DayOfWeek.SATURDAY));
      System.out.println("Second Saturday on : " + secondSaturday);
   }
}

验证结果

使用javac编译器编译类,,如下所示 −

C:\JAVA>javac Java8Tester.java

现在按如下方式运行 Java8Tester −

C:\JAVA>java Java8Tester

它应该产生以下结果 −

Current date: 2014-12-10
Next Tuesday on : 2014-12-16
Second Saturday on : 2014-12-13

向后兼容性

一个 toInstant() 方法被添加到原始的 Date 和 Calendar 对象,可用于将它们转换为新的 Date-Time API。 使用 ofInstant(Insant,ZoneId) 方法获取 LocalDateTime 或 ZonedDateTime 对象。 让我们看看它们的实际效果。

使用您选择的任何编辑器(例如 C:\> JAVA)创建以下 Java 程序。

Java8Tester.java

import java.time.LocalDateTime;
import java.time.ZonedDateTime;

import java.util.Date;

import java.time.Instant;
import java.time.ZoneId;

public class Java8Tester {

   public static void main(String args[]) {
      Java8Tester java8tester = new Java8Tester();
      java8tester.testBackwardCompatability();
   }
	
   public void testBackwardCompatability() {
      //获取当前日期
      Date currentDate = new Date();
      System.out.println("Current date: " + currentDate);
		
      //以毫秒为单位获取当前日期的瞬间
      Instant now = currentDate.toInstant();
      ZoneId currentZone = ZoneId.systemDefault();
		
      LocalDateTime localDateTime = LocalDateTime.ofInstant(now, currentZone);
      System.out.println("Local date: " + localDateTime);
		
      ZonedDateTime zonedDateTime = ZonedDateTime.ofInstant(now, currentZone);
      System.out.println("Zoned date: " + zonedDateTime);
   }
}

验证结果

使用javac编译器编译类,,如下所示 −

C:\JAVA>javac Java8Tester.java

现在按如下方式运行 Java8Tester −

C:\JAVA>java Java8Tester

它应该产生以下输出 −

Current date: Wed Dec 10 05:44:06 UTC 2014
Local date: 2014-12-10T05:44:06.635
Zoned date: 2014-12-10T05:44:06.635Z[Etc/UTC]