Java 8新的日期/时间API


使用Java 8,引入了新的Date-Time API,以弥补旧Date-time API的以下缺点。

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

  • 设计不良:默认日期从1900开始,月份从1开始,日期从0开始,因此没有统一性。旧的API具有较少直接的日期操作方法。新的API为此类操作提供了许多实用程序方法。

  • 时区处理困难:开发人员必须编写大量代码来解决时区问题。开发新的API时要牢记特定于领域的设计。

Java 8在包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//获取当前日期和时间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//获取当前日期和时间30

分区日期时间API


当考虑时区时,将使用分区的日期时间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//获取当前日期和时间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//获取当前日期和时间30+05:00[Asia/Karachi]
ZoneId: Europe/Paris
CurrentZone: Etc/UTC

计时单位枚举


Java 8中添加了java.time.temporal.ChronoUnit枚举,以替换旧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

期限和期限


使用Java 8,引入了两个专门的类来处理时间差。

  • 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使用编译类06 UTC 2014
Local date: 2014-12-10T05使用编译类06.635
Zoned date: 2014-12-10T05使用编译类06.635Z[Etc/UTC]