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]