添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

Java8时间日期处理新特性

伴随lambda表达式、streams以及一系列小优化,Java 8 推出了全新的日期时间API。Java处理日期、日历和时间的不足之处:将 java.util.Date 设定为可变类型,以及 SimpleDateFormat 的非线程安全使其应用非常受限。然后就在 java8 上面增加新的特性。全新API的众多好处之一就是,明确了日期时间概念,例如:瞬时(instant)、长短(duration)、日期、时间、时区和周期。同时继承了Joda 库按人类语言和计算机各自解析的时间处理方式。不同于老版本,新API基于ISO标准日历系统,java.time包下的所有类都是不可变类型而且线程安全。

Instant:瞬时实例。

LocalDate:本地日期,不包含具体时间 例如:2014-01-14 可以用来记录生日、纪念日、加盟日等。

LocalTime:本地时间,不包含日期。

LocalDateTime:组合了日期和时间,但不包含时差和时区信息。

ZonedDateTime:最完整的日期时间,包含时区和相对UTC或格林威治的时差

获取当前时间的对象

  1. LocalDateTime localDateTime = LocalDateTime.now();
  2. Date date = new Date ();

localDateTime 相比 Date 更像是一个工具类,就是为了时间操作使用。其构造方法是私有的

从字符串中解析

字符串 2019-01-11 解析成时间对象

  1. String str = "2019-01-11" ;
  2. DateTimeFormatter formatter = DateTimeFormatter.ofPattern( "yyyy-MM-dd" );
  3. LocalDate localDate = LocalDate.parse(str, formatter);
  4. SimpleDateFormat simpleDateFormat = new SimpleDateFormat( "yyyy-MM-dd" );
  5. try {
  6. Date date = simpleDateFormat.parse(str);
  7. } catch (ParseException e) {
  8. e.printStackTrace();

DateTimeFormatter的包路径是java.time.format和LocalDate一样在java.time下面,而SimpleDateFormat和Date是不同的。所以当判断引入路径的时候更容易判断。
当解析失败的时候,两个异常的抛出不一样,DateTimeFormatter抛出的是DateTimeParseException,继承自RuntimeException,而ParseException明显继承的是Exception。
个人感觉这个思路是,前者如果抛出异常那就是编程上错误,而后者则是的程序代码的不稳定性。更倾向于第一种的异常设计,应该加强对入参的检测判断,而不是通过捕获异常去处理入参的错误。(类似NumberFormatException)

LocalDate比Date更强的初始化时间

Date 设置某个日期,基本上3个方式,时间戳/Calendar/字符串解析。相对的LocalDate就简单很多

  1. LocalDate .of (2019,1,12);

时间戳的转换

LocalDate 时间戳的转换不如 Date 直接。主要因为 LocalDate 本身是没有时区的。

时间戳传LocalDateTime

  1. long timestamp = System.currentTimeMillis();
  2. Instant instant = Instant.ofEpochMilli(timestamp);
  3. LocalDateTime.ofInstant(instant, ZoneId.systemDefault());

LocalDateTime 转时间戳

  1. LocalDateTime dateTime = LocalDateTime.now();
  2. dateTime .toInstant(ZoneOffset.ofHours(8)).toEpochMilli();
  3. dateTime .toInstant(ZoneOffset.of( "+08:00" )).toEpochMilli();
  4. dateTime .atZone(ZoneId.systemDefault()).toInstant().toEpochMilli();

和Date互转

  1. import java.time.Instant;
  2. import java.util.Date;
  3. public class Main {
  4. public static void main( String [] args) {
  5. Date dt = new Date ();
  6. System.out.println( "Date: " + dt);
  7. Instant in = dt.toInstant();
  8. System.out.println( "Instant: " + in );
  9. Date dt2  = Date .from( in );
  10. System.out.println( "Date: " + dt2);

更好的理解和操作方式

Date、Calendar的操作,例如设置月份,day of week 的使用都不符合常规,例如1月的定义是0,周一是0。1号是0,相对于的 LocalDate就比较友好。例如DayOfWeek是枚举类型能更直观的使用

另外日期和时间的计算 可以直接进行加减和比较, 使用 ' ' 的示例:

Calendar 操作日期则是使用 get set 方法很不友好如:

Calendar cal = Calendar.getInstance();

cal.set(Calendar.MONTH, cal.get(Calendar.MONTH) + 1)

线程安全性比较

Java8提供的日期处理API LocalDate 等系列都是线程安全的,每一个字段都用了 final 关键字了,所以进行操作后都是返回新的copy对象

Date 线程不安全, get,set 在多线程的时候容易出现问题,不过 set 方法已经标注 @Deprecated 废弃了。当然不是因为线程安全问题废弃的,是因为有了更好的替代

SimpleDateFormat的线程安全性

在一定负载情况下, SimpleDateFormat 会出问题的。当然 SimpleDateFormat 线程不安全应该人尽皆知的,但依然有不安全的使用,但每次使用都 new 一个实例,当负载大的时候也不好。

Java8时间常用计算实例

  1. import java.time.Clock;
  2. import java.time.Instant;
  3. import java.time.LocalDate;
  4. import java.time.LocalDateTime;
  5. import java.time.LocalTime;
  6. import java.time.Month;
  7. import java.time.MonthDay;
  8. import java.time.OffsetDateTime;
  9. import java.time.ZoneOffset;
  10. import java.time.Period;
  11. import java.time.YearMonth;
  12. import java.time.ZoneId;
  13. import java.time.ZonedDateTime;
  14. import java.time.format.DateTimeFormatter;
  15. import java.time.temporal.ChronoUnit;
  16. import java.util.Date;
  17. public class DateTest {
  18. // 获取今天的日期
  19. public void getCurrentDate (){
  20. LocalDate today = LocalDate.now();
  21. System. out .println( "Today's Local date : " + today);
  22. // 这个是作为对比
  23. Date date = new Date();
  24. System. out .println(date);
  25. // 获取年、月、日信息
  26. public void getDetailDate (){
  27. LocalDate today = LocalDate.now();
  28. int year = today.getYear();
  29. int month = today.getMonthValue();
  30. int day = today.getDayOfMonth();
  31. System. out .printf( "Year : %d  Month : %d  day : %d t %n" , year, month, day);
  32. // 处理特定日期
  33. public void handleSpecilDate (){
  34. LocalDate dateOfBirth = LocalDate.of( 2018 , 01 , 21 );
  35. System. out .println( "The specil date is : " + dateOfBirth);
  36. // 判断两个日期是否相等
  37. public void compareDate (){
  38. LocalDate today = LocalDate.now();
  39. LocalDate date1 = LocalDate.of( 2018 , 01 , 21 );
  40. if (date1. equals (today)){
  41. System. out .printf( "TODAY %s and DATE1 %s are same date %n" , today, date1);
  42. // 处理周期性的日期
  43. public void cycleDate (){
  44. LocalDate today = LocalDate.now();
  45. LocalDate dateOfBirth = LocalDate.of( 2018 , 01 , 21 );
  46. MonthDay birthday = MonthDay.of(dateOfBirth.getMonth(), dateOfBirth.getDayOfMonth());
  47. MonthDay currentMonthDay = MonthDay. from (today);
  48. if (currentMonthDay. equals (birthday)){
  49. System. out .println( "Many Many happy returns of the day !!" );
  50. } else {
  51. System. out .println( "Sorry, today is not your birthday" );
  52. // 获取当前时间
  53. public void getCurrentTime (){
  54. LocalTime time = LocalTime.now();
  55. System. out .println( "local time now : " + time);
  56. // 增加小时
  57. public void plusHours (){
  58. LocalTime time = LocalTime.now();
  59. LocalTime newTime = time.plusHours( 2 ); // 增加两小时
  60. System. out .println( "Time after 2 hours : " +  newTime);
  61. // 如何计算一周后的日期
  62. public void nextWeek (){
  63. LocalDate today = LocalDate.now();
  64. LocalDate nextWeek = today.plus( 1 , ChronoUnit.WEEKS);
  65. System. out .println( "Today is : " + today);
  66. System. out .println( "Date after 1 week : " + nextWeek);
  67. // 计算一年前或一年后的日期
  68. public void minusDate (){
  69. LocalDate today = LocalDate.now();
  70. LocalDate previousYear = today.minus( 1 , ChronoUnit.YEARS);
  71. System. out .println( "Date before 1 year : " + previousYear);
  72. LocalDate nextYear = today.plus( 1 , ChronoUnit.YEARS);
  73. System. out .println( "Date after 1 year : " + nextYear);
  74. public void clock (){
  75. // 根据系统时间返回当前时间并设置为 UTC
  76. Clock clock = Clock.systemUTC();
  77. System. out .println( "Clock : " + clock);
  78. // 根据系统时钟区域返回时间
  79. Clock defaultClock = Clock.systemDefaultZone();
  80. System. out .println( "Clock : " + clock);
  81. // 如何用 Java 判断日期是早于还是晚于另一个日期
  82. public void isBeforeOrIsAfter (){
  83. LocalDate today = LocalDate.now();
  84. LocalDate tomorrow = LocalDate.of( 2018 , 1 , 29 );
  85. if (tomorrow.isAfter(today)){
  86. System. out .println( "Tomorrow comes after today" );
  87. LocalDate yesterday = today.minus( 1 , ChronoUnit.DAYS);
  88. if (yesterday.isBefore(today)){
  89. System. out .println( "Yesterday is day before today" );
  90. // 时区处理
  91. public void getZoneTime (){
  92. // 设置时区
  93. ZoneId america = ZoneId.of( "America/New_York" );
  94. LocalDateTime localtDateAndTime = LocalDateTime.now();
  95. ZonedDateTime dateAndTimeInNewYork  = ZonedDateTime.of(localtDateAndTime, america );
  96. System. out .println( " 现在的日期和时间在特定的时区 : " + dateAndTimeInNewYork);
  97. // 使用 YearMonth 类处理特定的日期
  98. public void checkCardExpiry (){
  99. YearMonth currentYearMonth = YearMonth.now();
  100. System. out .printf( "Days in month year %s: %d%n" , currentYearMonth, currentYearMonth.lengthOfMonth());
  101. YearMonth creditCardExpiry = YearMonth.of( 2028 , Month.FEBRUARY);
  102. System. out .printf( "Your credit card expires on %s %n" , creditCardExpiry);
  103. // 检查闰年
  104. public void isLeapYear (){
  105. LocalDate today = LocalDate.now();
  106. if (today.isLeapYear()){
  107. System. out .println( "This year is Leap year" );
  108. } else {
  109. System. out .println( "2018 is not a Leap year" );
  110. // 计算两个日期之间的天数和月数
  111. public void calcDateDays (){
  112. LocalDate today = LocalDate.now();
  113. LocalDate java8Release = LocalDate.of( 2018 , Month.MAY, 14 );
  114. Period periodToNextJavaRelease = Period.between(today, java8Release);
  115. System. out .println( "Months left between today and Java 8 release : " + periodToNextJavaRelease.getMonths() );
  116. // 包含时差信息的日期和时间
  117. public void ZoneOffset (){
  118. LocalDateTime datetime = LocalDateTime.of( 2018 , Month.FEBRUARY, 14 , 19 , 30 );
  119. ZoneOffset offset = ZoneOffset.of( "+05:30" );
  120. OffsetDateTime date = OffsetDateTime.of(datetime, offset);
  121. System. out .println( "Date and Time with timezone offset in Java : " + date);
  122. // 获取时间戳
  123. public void getTimestamp (){
  124. Instant timestamp = Instant.now();
  125. System. out .println( "What is value of this instant " + timestamp);
  126. // 使用预定义的格式化工具去解析或格式化日期
  127. public void formateDate (){
  128. String dayAfterTommorrow = "20180210" ;
  129. LocalDate formatted = LocalDate.parse(dayAfterTommorrow, DateTimeFormatter.BASIC_ISO_DATE);
  130. System. out .printf( "Date generated from String %s is %s %n" , dayAfterTommorrow, formatted);
  131. public static void main (String[] args) {
  132. DateTest dt = new DateTest();
  133. dt.formateDate();

    Java 8日期时间API的重点

    1)提供了javax.time.ZoneId 获取时区。

    2)提供了LocalDate和LocalTime类。

    3)Java 8 的所有日期和时间API都是不可变类并且线程安全,而现有的Date和Calendar API中的java.util.Date和SimpleDateFormat是非线程安全的。

    4)主包是 java.time,包含了表示日期、时间、时间间隔的一些类。里面有两个子包java.time.format用于格式化, java.time.temporal用于更底层的操作。

    5)时区代表了地球上某个区域内普遍使用的标准时间。每个时区都有一个代号,格式通常由区域/城市构成(Asia/Tokyo),在加上与格林威治或 UTC的时差。例如:东京的时差是+09:00。