zoukankan      html  css  js  c++  java
  • Java 8新特性之Date/Time(八恶人-4)

     Mannix‘s Marauders -Chris Mannix  曼尼克斯掠夺者 曼尼克斯·克里斯

    “I'm the new sheriff of Red Rock.”  “我是红石镇的新任警长”

    一、基础介绍

     

    目前Java8已经实现了JSR310的全部内容。新增了java.time包定义的类表示了日期-时间概念的规则,包括instants, durations, dates, times, time-zones and periods。这些都是基于ISO日历系统,它又是遵循 Gregorian规则的。最重要的一点是值不可变,且线程安全,通过下面一张图,我们快速看下java.time包下的一些主要的类的值的格式,方便理解。

    Java 8通过发布新的Date-Time API (JSR 310)来进一步加强对日期与时间的处理。对日期与时间的操作一直是Java程序员最痛苦的地方之一。标准的 java.util.Date以及后来的java.util.Calendar一点没有改善这种情况(可以这么说,它们一定程度上更加复杂)。

    这种情况直接导致了Joda-Time——一个可替换标准日期/时间处理且功能非常强大的Java API的诞生。Java 8新的Date-Time API (JSR 310)在很大程度上受到Joda-Time的影响,并且吸取了其精髓。新的java.time包涵盖了所有处理日期,时间,日期/时间,时区,时刻(instants),过程(during)与时钟(clock)的操作。在设计新版API时,十分注重与旧版API的兼容性:不允许有任何的改变(从java.util.Calendar中得到的深刻教训)。如果需要修改,会返回这个类的一个新实例。

     Java8中的时间函数包已经重新在java.time中安家。

     

    二、具体使用

    1、获取UTC

    让我们用例子来看一下新版API主要类的使用方法。第一个是Clock类,它通过指定一个时区,然后就可以获取到当前的时刻,日期与时间。Clock可以替换System.currentTimeMillis()与TimeZone.getDefault()。

    获取到的时间是UTC,和北京时间比这个早8个小时。如果结果是上午6点,对应的北京时间是下午两点(在小时的地方+8)

    // Get the system clock as UTC offset 
    final Clock clock = Clock.systemUTC();
    System.out.println( clock.instant() );

    下面是程序在控制台上的输出:

    2017-12-22T07:47:18.124Z

     

    2、获取当地日期、时间

    我们需要关注的其他类是LocaleDate与LocalTime。LocaleDate只持有ISO-8601格式且无时区信息的日期部分。相应的,LocaleTime只持有ISO-8601格式且无时区信息的时间部分。LocaleDate与LocalTime都可以从Clock中得到。

    LocaleDate与LocalTim获取到的是本时区的时间//使用的场合最多,可以使用.now()直接获取,也可以使用of方法进行构造

    // Get the local date and local time
    final LocalDate date = LocalDate.now();
    System.out.println( date );

    // Get the local date and local time
    final LocalTime time = LocalTime.now();
    System.out.println( time );

    //Creating LocalDate by providing input arguments
    LocalDate firstDay_2014 = LocalDate.of(2014, 1, 1);
    System.out.println("Specific Date="+firstDay_2014);

    //Creating LocalTime by providing input arguments
    LocalTime specificTime = LocalTime.of(12,20,25,40);
    System.out.println("Specific Time of Day="+specificTime);

    下面是程序在控制台上的输出:

    2017-12-22
    15:46:32.255
    Specific Date=2014-01-01
    Specific Time of Day=12:20:25.000000040

     

    3、获取当地日期和时间

    LocaleDateTime把LocaleDate与LocaleTime的功能合并起来,它持有的是ISO-8601格式无时区信息的日期与时间。同样可以使用of进行构造

    // Get the local date/time
    final LocalDateTime datetime = LocalDateTime.now();
    System.out.println( datetime );

    下面是程序在控制台上的输出:

    2014-04-12T11:37:52.309
    2014-04-12T11:37:52.309
    

    4、指定时区的日期时间获取

    如果你需要特定时区的日期/时间,那么ZonedDateTime是你的选择。它持有ISO-8601格式具具有时区信息的日期与时间。下面是一些不同时区的例子:

    // Get the zoned date/time
    final ZonedDateTime zonedDatetime = ZonedDateTime.now();
    final ZonedDateTime zonedDatetimeFromClock = ZonedDateTime.now( clock );
    final ZonedDateTime zonedDatetimeFromZone = ZonedDateTime.now( ZoneId.of( "America/Los_Angeles" ) );
             
    System.out.println( zonedDatetime );
    System.out.println( zonedDatetimeFromClock );
    System.out.println( zonedDatetimeFromZone );

    下面是程序在控制台上的输出:

    2014-04-12T11:47:01.017-04:00[America/New_York]
    2014-04-12T15:47:01.017Z
    2014-04-12T08:47:01.017-07:00[America/Los_Angeles]

     

    5、计算日期时间间隔

    使用Duration类:在秒与纳秒级别上的一段时间。Duration使计算两个日期间的不同变的十分简单。下面让我们看一个这方面的例子。

    // Get duration between two dates
    final LocalDateTime from = LocalDateTime.of( 2014, Month.APRIL, 16, 0, 0, 0 );
    final LocalDateTime to = LocalDateTime.of( 2015, Month.APRIL, 16, 23, 59, 59 );
     
    final Duration duration = Duration.between( from, to );
    System.out.println( "Duration in days: " + duration.toDays() );
    System.out.println( "Duration in hours: " + duration.toHours() );
    

      

    上面的例子计算了两个日期2014年4月16号与2014年4月16号之间的过程。下面是程序在控制台上的输出:

    Duration in days: 365
    Duration in hours: 8783

     

    6、日期时间的加减操作

    package com.journaldev.java8.time;
     
    import java.time.LocalDate;
    import java.time.LocalTime;
    import java.time.Period;
    import java.time.temporal.TemporalAdjusters;
     
    public class DateAPIUtilities {
     
        public static void main(String[] args) {
        LocalDate today = LocalDate.now();

        LocalTime nowTime = LocalTime.now();


        //Get the Year, check if it's leap year
        System.out.println("Year "+today.getYear()+" is Leap Year? "+today.isLeapYear());

        //Compare two LocalDate for before and after
        System.out.println("Today is before 01/01/2015? "+today.isBefore(LocalDate.of(2015,1,1)));

        //Create LocalDateTime from LocalDate
        System.out.println("Current Time="+today.atTime(LocalTime.now()));

        //plus and minus operations
        System.out.println("10 days after today will be "+today.plusDays(10));
        System.out.println("3 weeks after today will be "+today.plusWeeks(3));
        System.out.println("20 months after today will be "+today.plusMonths(20));
        System.out.println("20 years after today will be "+today.plusYears(20));

        System.out.println("10 days before today will be "+today.minusDays(10));
        System.out.println("3 weeks before today will be "+today.minusWeeks(3));
        System.out.println("20 months before today will be "+today.minusMonths(20));

        System.out.println("20 Hours after now will be "+nowTime.plusHours(20));
        System.out.println("20 Hours before now will be "+nowTime.minusHours(20));
    } }

      输出结果是

    Year 2017 is Leap Year? false
    Today is before 01/01/2015? false
    Current Time=2017-12-22T16:05:46.767
    10 days after today will be 2018-01-01
    3 weeks after today will be 2018-01-12
    20 months after today will be 2019-08-22
    20 years after today will be 2037-12-22
    10 days before today will be 2017-12-12
    3 weeks before today will be 2017-12-01
    20 months before today will be 2016-04-22
    20 Hours after now will be 12:05:46.767
    20 Hours before now will be 20:05:46.767

     

    7、Date/Time和String转换

    再来个很常用的功能,String和Date的转换//使用的场合最多

    DateTimeFormatter f = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
    //时间转为字符串
    LocalDateTime date =LocalDateTime.now();
    String str = date.format(f);  
    String str = date.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");//直接把转换格式写在参数里面,相当于两行代码
    //字符串转为时间
    date = LocalDateTime.parse(“2017-12-22 16:18:32”,f); 
    date = LocalDateTime.parse(“2017-12-22 16:18:32”,DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");//直接把转换格式写在参数里面,相当于两行代码
    );
     

      

     

     三、功能整合列表

     

    public class TimeIntroduction {
        public static void testClock() throws InterruptedException {
            //时钟提供给我们用于访问某个特定 时区的 瞬时时间、日期 和 时间的。  
            Clock c1 = Clock.systemUTC(); //系统默认UTC时钟(当前瞬时时间 System.currentTimeMillis())  
            System.out.println(c1.millis()); //每次调用将返回当前瞬时时间(UTC)  
            Clock c2 = Clock.systemDefaultZone(); //系统默认时区时钟(当前瞬时时间)  
            Clock c31 = Clock.system(ZoneId.of("Europe/Paris")); //巴黎时区  
            System.out.println(c31.millis()); //每次调用将返回当前瞬时时间(UTC)  
            Clock c32 = Clock.system(ZoneId.of("Asia/Shanghai"));//上海时区  
            System.out.println(c32.millis());//每次调用将返回当前瞬时时间(UTC)  
            Clock c4 = Clock.fixed(Instant.now(), ZoneId.of("Asia/Shanghai"));//固定上海时区时钟  
            System.out.println(c4.millis());
            Thread.sleep(1000);
            System.out.println(c4.millis()); //不变 即时钟时钟在那一个点不动  
            Clock c5 = Clock.offset(c1, Duration.ofSeconds(2)); //相对于系统默认时钟两秒的时钟  
            System.out.println(c1.millis());
            System.out.println(c5.millis());
        }
        public static void testInstant() {
            //瞬时时间 相当于以前的System.currentTimeMillis()  
            Instant instant1 = Instant.now();
            System.out.println(instant1.getEpochSecond());//精确到秒 得到相对于1970-01-01 00:00:00 UTC的一个时间  
            System.out.println(instant1.toEpochMilli()); //精确到毫秒  
            Clock clock1 = Clock.systemUTC(); //获取系统UTC默认时钟  
            Instant instant2 = Instant.now(clock1);//得到时钟的瞬时时间  
            System.out.println(instant2.toEpochMilli());
            Clock clock2 = Clock.fixed(instant1, ZoneId.systemDefault()); //固定瞬时时间时钟  
            Instant instant3 = Instant.now(clock2);//得到时钟的瞬时时间  
            System.out.println(instant3.toEpochMilli());//equals instant1  
        }
        public static void testLocalDateTime() {
            //使用默认时区时钟瞬时时间创建 Clock.systemDefaultZone() -->即相对于 ZoneId.systemDefault()默认时区  
            LocalDateTime now = LocalDateTime.now();
            System.out.println(now);
    //自定义时区  
            LocalDateTime now2 = LocalDateTime.now(ZoneId.of("Europe/Paris"));
            System.out.println(now2);//会以相应的时区显示日期  
    //自定义时钟  
            Clock clock = Clock.system(ZoneId.of("Asia/Dhaka"));
            LocalDateTime now3 = LocalDateTime.now(clock);
            System.out.println(now3);//会以相应的时区显示日期  
    //不需要写什么相对时间 如java.util.Date 年是相对于1900 月是从0开始  
    //2013-12-31 23:59  
            LocalDateTime d1 = LocalDateTime.of(2013, 12, 31, 23, 59);
    //年月日 时分秒 纳秒  
            LocalDateTime d2 = LocalDateTime.of(2013, 12, 31, 23, 59, 59, 11);
    //使用瞬时时间 + 时区  
            Instant instant = Instant.now();
            LocalDateTime d3 = LocalDateTime.ofInstant(Instant.now(), ZoneId.systemDefault());
            System.out.println(d3);
    //解析String--->LocalDateTime  
            LocalDateTime d4 = LocalDateTime.parse("2013-12-31T23:59");
            System.out.println(d4);
            LocalDateTime d5 = LocalDateTime.parse("2013-12-31T23:59:59.999");//999毫秒 等价于999000000纳秒  
            System.out.println(d5);
    //使用DateTimeFormatter API 解析 和 格式化  
            DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss");
            LocalDateTime d6 = LocalDateTime.parse("2013/12/31 23:59:59", formatter);
            System.out.println(formatter.format(d6));
    //时间获取  
            System.out.println(d6.getYear());
            System.out.println(d6.getMonth());
            System.out.println(d6.getDayOfYear());
            System.out.println(d6.getDayOfMonth());
            System.out.println(d6.getDayOfWeek());
            System.out.println(d6.getHour());
            System.out.println(d6.getMinute());
            System.out.println(d6.getSecond());
            System.out.println(d6.getNano());
    //时间增减  
            LocalDateTime d7 = d6.minusDays(1);
            LocalDateTime d8 = d7.plus(1, IsoFields.QUARTER_YEARS);
    //LocalDate 即年月日 无时分秒  
    //LocalTime即时分秒 无年月日  
    //API和LocalDateTime类似就不演示了  
        }
        public static void testZonedDateTime() {
            //即带有时区的date-time 存储纳秒、时区和时差(避免与本地date-time歧义)。  
    //API和LocalDateTime类似,只是多了时差(如2013-12-20T10:35:50.711+08:00[Asia/Shanghai])  
            ZonedDateTime now = ZonedDateTime.now();
            System.out.println(now);
            ZonedDateTime now2 = ZonedDateTime.now(ZoneId.of("Europe/Paris"));
            System.out.println(now2);
    //其他的用法也是类似的 就不介绍了  
            ZonedDateTime z1 = ZonedDateTime.parse("2013-12-31T23:59:59Z[Europe/Paris]");
            System.out.println(z1);
        }
        public static void testDuration() {
            //表示两个瞬时时间的时间段  
            Duration d1 = Duration.between(Instant.ofEpochMilli(System.currentTimeMillis() - 12323123), Instant.now());
    //得到相应的时差  
            System.out.println(d1.toDays());
            System.out.println(d1.toHours());
            System.out.println(d1.toMinutes());
            System.out.println(d1.toMillis());
            System.out.println(d1.toNanos());
    //1天时差 类似的还有如ofHours()  
            Duration d2 = Duration.ofDays(1);
            System.out.println(d2.toDays());
        }
        public static void testChronology() {
            //提供对java.util.Calendar的替换,提供对年历系统的支持  
            Chronology c = HijrahChronology.INSTANCE;
            ChronoLocalDateTime d = c.localDateTime(LocalDateTime.now());
            System.out.println(d);
        }
        /**
         * 新旧日期转换
         */
        public static void testNewOldDateConversion(){
            Instant instant=new Date().toInstant();
            Date date=Date.from(instant);
            System.out.println(instant);
            System.out.println(date);
        }
        public static void main(String[] args) throws InterruptedException {
            testClock();
            testInstant();
            testLocalDateTime();
            testZonedDateTime();
            testDuration();
            testChronology();
            testNewOldDateConversion();
        }
    }

     

     

     

     

     参考链接:

    http://www.importnew.com/11908.html#methodReferences

    https://my.oschina.net/benhaile/blog/193956

    图片来源:八恶人(movie)

  • 相关阅读:
    网络相关配置
    RestTemplate 工具类以及拦截器配置打印调用日志
    Redis(2)九大数据类型及落地案例
    JUC(7)ThreadLocal
    JUC(5)原子类
    JUC(6)LockSupport
    JUC(4)Volatile
    JUC(3)Java内存模型JMM
    JUC(1)说说Java“锁”事
    JUC(2)线程中断机制
  • 原文地址:https://www.cnblogs.com/Jacck/p/8080856.html
Copyright © 2011-2022 走看看