新款时间日期API已上线

一世执手 提交于 2019-11-30 09:24:07

历史

   早期,Java 对时间和日期的使用往往重度依赖 java.util.Datejava.util.Calendar。 可惜的是,这2个api 本身不够健壮,有类似线程不安全等诸多问题,于是乎2000年左右,市面上出现了很多用于处理时间类的框架,如Joda-Time。 在Java 8 时代 官方为JDK开发新的时间/日期API的进程:JSR-310,启动了。于是就出现了新一套健壮的API。

 

Roadmap

新的API

Java 8  中 新的API 包括了 5个包

  • java.time

  • java.time.chrono

  • java.time.format

  • java.time.temporal

  • java.time.zone

    java.time 是基础包,里面包含了 时间 日期的 基础类 如  LocalDate, LocalTime等

    java.time.chrono  有点像linux 的守护进程 chronyd  用来访问不同的时间服务器。

    java.time.format 顾名思义用来格式化和转换日期时间

    java.time.temporal 可能也想不出好名字吧,里面都是些底层扩展类

    java.time.zone 用来对各时区的支持

 

1. 时间线

     Java8中 引入了时间线的概念,时间是连续不断,抽象成一条线,线上的点表示某一时刻。

一班来说,这个设定主要是面对机器的,不用关心具体的年月日时分秒,内部以秒或者纳秒表示。

1.1 时间点 Instant

      时间点是表示时间的某时某刻的,相对的它是时间线上的一点。

java.time包通过值类型Instant提供机器视图。

Instant表示时间线上的一点,而不需要任何上下文信息(时区,年份)

概念上讲,它只是简单的表示自1970年1月1日0时0分0秒(UTC)开始的秒数。

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

Instant start = Instant.now()
Instant end = Instant.now()

 

1.2 Period

    Period表示以年、月、日衡量的时长。例如,“3年2个月零6天”。表示的是一段时间 以日月年为单位.

在日期的操作中 可以作为参数来作为日期的加减运算。

Period period = Period.now()

1.3 Duration

    Duration表示以秒和纳秒为基准的时长。例如,“23.6秒”。Duration表示的是一段时间 以 分十秒为单位. 在时间的操作中 可以作为参数 来作为时间的加减运算。

Duration duration = Duration.()System..println(duration)System..println(duration.getSeconds())

 

--------------------------TBD-----------------------

 

2. 不变模式  Immutable

    新的API 提倡采用不变模式,虽然时间是一直在变化的,某事某刻的表示它是固定的,哪怕对日期进行加减,操作之后的日期则表示新的时刻。

   所以Java时间相关的API,在原始时间对象上 做相应的加减操作,原来的日期对象不变,会产生一个新的对象。

3. 本地时间

3.1 本地日期 LocalDate

 LocalDate 是新API 中 最基础也是最重要的类, 它是一个 不可变的类(final 修饰),本地日期只是一个术语,用来表示一个具体日期,不同地区的日期可能不同 故采用 local 这个术语,其实它是时区无关的,这个类没有包含任何时区信息,就简单的认为一个日期即可。

3.1.1 日期的创建

   日期的创建 采用了工厂方法的模式,所有的创建方法都是私有的,常用的是调用对外开放的方法of 来生成一个 日期实例

LocalDate localDate = LocalDate.of(int year, Month month, int dayOfMonth);
LocalDate localDate = LocalDate.of(int year, int month, int dayOfMonth);

 

3.1.2 日期的常用方法 - 读

 

    LocalDate 的toString方法 默认返回的日期格式  采用 ISO-8601标准日期  即   YYYY-MM-DD

System.out.println(localDate); //2016-02-18

   LocalDate 还提供了其他很多常用方法来获得日期的相关值

System.out.println(localDate.getMonth());       // Month FEBRUARY 当前月
System.out.println(localDate.getMonthValue());  // int  1 当前月的数值

System.out.println(localDate.getDayOfMonth());  // int 18 当前月第X天
System.out.println(localDate.getDayOfWeek());   // DayOfWeek  Monday 当前周的天
System.out.println(localDate.getDayOfYear());   // int 49 当前年的第X天

System.out.println(localDate.getYear());        // int 2016 当前年

System.out.println(localDate.lengthOfYear());   // int 366 当前年共有多少天
System.out.println(localDate.lengthOfMonth());  // int 29  当前月共有多少天

3.1.3 日期的常用方法 - 操作

LocalDate 是一个不可变的类,它的每一次操作 都会伴随生成一个新类

localDate.withYear()
localDate.plusMonths()
localDate.minusDays())

3.1.4 日期的常用方法 - 属性判断

LocalDate 也提供了一些属性判断的功能

 localDate.isLeapYear(); // true (是闰年)

 

3.2 本地时间 LocalTime

    LocalTime 的用法和 LocalDate 类似同样与与时区无关,  它的默认格式也是采用 ISO-8601 格式为 HH:mm:ss

时间的创建

LocalTime localTime = LocalTime.of(int hour, int minute, int second);
LocalTime localTime = LocalTime.of(int hour, int minute, int second, int nanoOfSecond)
LocalTime localTime = LocalTime.of(int hour, int minute);

例子:

LocalTime localTime = LocalTime.of(17,10,58);

3.3 本地日期时间的组合 LocalDateTime

TBD

 

-------------------------------------Example TBD------------------------------

 

4 时区时间

时区 Zone

5 时间工具

5.1 时间计算器 ChrononUnit

ChronoUnit类可用于在单个时间单位内测量一段时间,例如天数或秒。

LocalDate startDate = LocalDate.of(1993, Month.OCTOBER, 19);
System.out.println("开始时间  : " + startDate);

LocalDate endDate = LocalDate.of(2017, Month.JUNE, 16);
System.out.println("结束时间 : " + endDate);

long daysDiff = ChronoUnit.DAYS.between(startDate, endDate);
System.out.println("两天之间的差在天数   : " + daysDiff);

5.2 格式转换器 DateTimeFormatter

    Java 8 中 java.time.format 中 围绕  DateTimeFormatter 构建了时间的解析与转换。Java 8 默认的时间显示格式是,ISO 8601标准格式。

5.2.1 使用格式转换器

采用DateTimeFormatter.format() 方法进行格式化输出。

首先需要实例化一个具体对应的时间格式转换器。

如要格式化LocalDate 则需要实例化一个针对LocalDate的转换器

需要注意的是转化器与日期时间对象要一一对应切不可 

LocalDate对象 用DateTimeFormatter.ofLocalDateTime.

这样会抛出UnsupportedTemporalyException.

在实例化一个转换器 需要传递一个格式参数,Java8提供了 预定义的格式和自定义声明

DateTimeFormatter shortDateTime = DateTimeFormatter.ofLocalizedDate(FormatStyle.MEDIUM);
DateTimeFormatter shortDateTime = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.MEDIUM);
DateTimeFormatter shortDateTime = DateTimeFormatter.ofLocalizedTime(FormatStyle.MEDIUM);

5.2.1.1 预定格式 FormatStyle

FormatSyle 是一个枚举类,它包括了Full,LONG,MEDIUM,SHORT几种常规的格式。

FormatStyle.FULL  // 'Tuesday, April 12, 1952 AD' or '3:30:42pm PST'.
FormatStyle.LONG // 'January 12, 1952'
FormatStyle.MEDIUM // 'Jan 12, 1952'.
FormatStyle.SHORT //'12.13.52' or '3:30pm'.

5.2.1.2 自定义格式 Pattern

除了之前介绍的一些预定义格式以外,还可以输出一些自定义的时间格式。采用字符串匹配模式

DateTimeFormatter f = DateTimeFormatter.ofPattern("MMMM dd, yyyy, hh:mm");

规则:

一般采用以下规则
 * MMMM 对应月份
   M outputs 1, 
   MM outputs 01, 
   MMM outputs Jan,
   MMMM outputs January.

 * dd,DD 对应天数
 * YYYY,YY,yyyy,yy 对应年份
 * HH 对应 24小时制的 小时
 * hh 对应 12小时制的 小时
 * mm 对应 分钟
 * ss 对应 秒
 * SS 对应 毫秒

此外 还可以包括 类似 / , : - 等分隔符

DateTimeFormatter f1 = DateTimeFormatter.ofPattern("MMMM dd, yyyy, hh:mm");
System.out.println(dateTime.format(f1)); // January 20, 2020, 01:12
DateTimeFormatter f2 = DateTimeFormatter.ofPattern("MM dd, yyyy, hh:mm");
System.out.println(dateTime.format(f2)); // 01 20, 2020, 01:12

DateTimeFormatter f3 = DateTimeFormatter.ofPattern("MMMM DD, YY, HH:mm");
System.out.println(dateTime.format(f3)); // January 20, 2020, 13:12
DateTimeFormatter f4 = DateTimeFormatter.ofPattern("MMMM dd, yyyy, hh:mm");
System.out.println(dateTime.format(f4)); // January 20, 2020, 01:12

DateTimeFormatter f5 = DateTimeFormatter.ofPattern("MMMM dd, yy, hh:mm:ss");
System.out.println(dateTime.format(f5)); // January 20, 2020, 01:12:34
DateTimeFormatter f6 = DateTimeFormatter.ofPattern("MMMM dd, yyyy, hh:mm:ss:SS");
System.out.println(dateTime.format(f6)); // January 20, 2020, 01:12:00

5.2.1.3 预留对象 DateTimeFormatter.IOS***

Java8 预先定义了一些常用格式的转换器,可以免去 自定义的麻烦.

DateTimeFormatter 内置了这些对象,可以直接调用。

DateTimeFormatter.ISO_LOCAL_DATE  //'2011-12-03'
DateTimeFormatter.ISO_OFFSET_DATE //'2011-12-03+01:00'
DateTimeFormatter.ISO_DATE        //'2011-12-03' or '2011-12-03+01:00'
DateTimeFormatter.ISO_LOCAL_TIME  //'10:15' or '10:15:30'
DateTimeFormatter.ISO_OFFSET_TIME //'10:15+01:00' or '10:15:30+01:00'
DateTimeFormatter.ISO_TIME        //'10:15', '10:15:30' or '10:15:30+01:00'
DateTimeFormatter.ISO_LOCAL_DATE_TIME //'2011-12-03T10:15:30'
DateTimeFormatter.ISO_OFFSET_DATE_TIME//'2011-12-03T10:15:30+01:00'
DateTimeFormatter.ISO_ZONED_DATE_TIME //'2011-12-03T10:15:30+01:00'

5.2.2 内置Format方法

    时间日期相关类自带一个formate(DateTimeFormatter.TYPE)方法,需要传递一个格式转换器,会根据格式转换器定义的格式做相应的转换。

//创建本地时间
LocalDate date = LocalDate.of(2020, Month.JANUARY, 20);
LocalTime time = LocalTime.of(11, 12, 34);
LocalDateTime dateTime = LocalDateTime.of(date, time);

date.format(DateTimeFormatter.ISO_LOCAL_DATE)); //2020-01-20
dateTime.format(DateTimeFormatter.ISO_LOCAL_DATE_TIME)); //2020-01-20T11:12:34
time.format(DateTimeFormatter.ISO_LOCAL_TIME)); //11:12:34

date.format(DateTimeFormatter.ISO_DATE); //2020-01-20
dateTime.format(DateTimeFormatter.ISO_DATE_TIME); //2020-01-20T11:12:34
time.format(DateTimeFormatter.ISO_TIME); //11:12:34

5.3 时间解析器 Parser

 时间的解析和DateTimeFormatter中 ofPatter 结合使用 

配合要解析的字符串对应 时间 就可以了。

DateTimeFormatter f = DateTimeFormatter.ofPattern("MM dd yyyy");
LocalDate date = LocalDate.parse("01 02 2015", f);
LocalTime time = LocalTime.parse("11:22");
System.out.println(date); // 2015-01-02
System.out.println(time); // 11:22

5.4

6 JDBC 与新日期

最新JDBC映射将把数据库的日期类型和Java 8的新类型关联起来:

SQL -> Java
--------------------------
date -> LocalDate
time -> LocalTime
timestamp -> LocalDateTime

 

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!