Java8中LocalDateTime的基本用法转载

2020年9月9日21:21:55来源:闲人鹤 2 23 8207字阅读27分21秒

简介

JDK8新特性里提供了3个时间类:LocalDateLocalTimeLocalDateTime

Java8中LocalDateTime的基本用法

在项目开发中,已经需要对Date类型进行格式,否则可读性很差,格式化Date类型要使用SimpleDateFormat,但SimpleDateFormat是线程不安全的。

为什么需要LocalDate、LocalTime、LocalDateTime

1.Date如果不格式化,打印出的日期可读性差。

  1. Tue Sep 10 09:34:04 CST 2019

2 使用SimpleDateFormat对时间进行格式化,但SimpleDateFormat是线程不安全的。SimpleDateFormat的format方法最终调用代码:

  1. private StringBuffer format(Date date, StringBuffer toAppendTo, FieldDelegate delegate)
  2. {
  3.     // Convert input date to time field list
  4.     calendar.setTime(date);
  5.     boolean useDateFormatSymbols = useDateFormatSymbols();
  6.     for(int i = 0; i < compiledPattern.length;)
  7.     {
  8.         int tag = compiledPattern[i] >>> 8;
  9.         int count = compiledPattern[i++] & 0xff;
  10.         if(count == 255)
  11.         {
  12.             count = compiledPattern[i++] << 16;
  13.             count |= compiledPattern[i++];
  14.         }
  15.         switch(tag)
  16.         {
  17.             case TAG_QUOTE_ASCII_CHAR:
  18.                 toAppendTo.append((char) count);
  19.                 break;
  20.             case TAG_QUOTE_CHARS:
  21.                 toAppendTo.append(compiledPattern, i, count);
  22.                 i += count;
  23.                 break;
  24.             default:
  25.                 subFormat(tag, count, delegate, toAppendTo, useDateFormatSymbols);
  26.                 break;
  27.         }
  28.     }
  29.     return toAppendTo;
  30. }

calendar是共享变量,并且这个共享变量没有做线程安全控制。当多个线程同时使用相同的SimpleDateFormat对象【如用static修饰的SimpleDateFormat】调用format方法时,多个线程会同时调用calendar.setTime方法,可能一个线程刚设置好time值 另外的一个线程马上把设置的time值给修改了导致返回的格式化时间可能是错误的。

在多并发情况下使用SimpleDateFormat需格外注意 SimpleDateFormat除了format是线程不安全以外,parse方法也是线程不安全的。parse方法实际调用alb.establish(calendar).getTime()方法来解析,alb.establish(calendar)方法里主要完成了。

  • 重置日期对象cal的属性值
  • 使用calb中中属性设置cal
  • 返回设置好的cal对象

但是这三步不是原子操作。

多线程并发如何保证线程安全 - 避免线程之间共享一个SimpleDateFormat对象,每个线程使用时都创建一次SimpleDateFormat对象 => 创建和销毁对象的开销大 - 对使用format和parse方法的地方进行加锁 => 线程阻塞性能差 - 使用ThreadLocal保证每个线程最多只创建一次SimpleDateFormat对象 => 较好的方法。Date对时间处理比较麻烦,比如想获取某年、某月、某星期,以及n天以后的时间,如果用Date来处理的话真是太难了,你可能会说Date类不是有getYear、getMonth这些方法吗,获取年月日很Easy,但都被弃用了。

Java8全新的日期和时间API

  • LocalDate

LocalDate是日期处理类,具体API如下:

  1. // 获取当前日期
  2. LocalDate now = LocalDate.now();
  3. // 设置日期
  4. LocalDate localDate = LocalDate.of(2019910);
  5. // 获取年
  6. int year = localDate.getYear(); //结果:2019
  7. int year1 = localDate.get(ChronoField.YEAR); //结果:2019
  8. // 获取月
  9. Month month = localDate.getMonth(); // 结果:SEPTEMBER
  10. int month1 = localDate.get(ChronoField.MONTH_OF_YEAR); //结果:9
  11. // 获取日
  12. int day = localDate.getDayOfMonth(); //结果:10
  13. int day1 = localDate.get(ChronoField.DAY_OF_MONTH); // 结果:10
  14. // 获取星期
  15. DayOfWeek dayOfWeek = localDate.getDayOfWeek(); //结果:TUESDAY
  16. int dayOfWeek1 = localDate.get(ChronoField.DAY_OF_WEEK); //结果:2
  • LocalTime

LocalTime是时间处理类,具体API如下:

  1. // 获取当前日期
  2. LocalDate now = LocalDate.now();
  3. // 设置日期
  4. LocalDate localDate = LocalDate.of(2019910);
  5. // 获取年
  6. int year = localDate.getYear(); //结果:2019
  7. int year1 = localDate.get(ChronoField.YEAR); //结果:2019
  8. // 获取月
  9. Month month = localDate.getMonth(); // 结果:SEPTEMBER
  10. int month1 = localDate.get(ChronoField.MONTH_OF_YEAR); //结果:9
  11. // 获取日
  12. int day = localDate.getDayOfMonth(); //结果:10
  13. int day1 = localDate.get(ChronoField.DAY_OF_MONTH); // 结果:10
  14. // 获取星期
  15. DayOfWeek dayOfWeek = localDate.getDayOfWeek(); //结果:TUESDAY
  16. int dayOfWeek1 = localDate.get(ChronoField.DAY_OF_WEEK); //结果:2
  • LocalDateTime

LocalDateTime可以设置年月日时分秒,相当于LocalDate + LocalTime

  1. // 获取当前日期时间
  2. LocalDateTime localDateTime = LocalDateTime.now();
  3. // 设置日期
  4. LocalDateTime localDateTime1 = LocalDateTime.of(2019, Month.SEPTEMBER, 10144656);
  5. LocalDateTime localDateTime2 = LocalDateTime.of(localDate, localTime);
  6. LocalDateTime localDateTime3 = localDate.atTime(localTime);
  7. LocalDateTime localDateTime4 = localTime.atDate(localDate);
  8. // 获取LocalDate
  9. LocalDate localDate2 = localDateTime.toLocalDate();
  10. // 获取LocalTime
  11. LocalTime localTime2 = localDateTime.toLocalTime();
  • Instant
  1. // 创建Instant对象
  2. Instant instant = Instant.now();
  3. // 获取秒
  4. long currentSecond = instant.getEpochSecond();
  5. // 获取毫秒
  6. long currentMilli = instant.toEpochMilli();
  • 修改LocalDate、LocalTime、LocalDateTime、Instant
  1. // 创建日期:2019-09-10 14:46:56
  2. LocalDateTime localDateTime = LocalDateTime.of(2019, Month.SEPTEMBER, 10144656);
  3. //增加一年
  4. localDateTime = localDateTime.plusYears(1); //结果: 2020-09-10 14:46:56
  5. localDateTime = localDateTime.plus(1, ChronoUnit.YEARS); //结果: 2021-09-10 14:46:56
  6. //减少一个月
  7. localDateTime = localDateTime.minusMonths(1); //结果: 2021-08-10 14:46:56
  8. localDateTime = localDateTime.minus(1, ChronoUnit.MONTHS); //结果: 2021-07-10 14:46:56

通过with修改某些值,年月日时分秒都可以通过with方法设置。

  1. //修改年为2019
  2. localDateTime = localDateTime.withYear(2020);
  3. //修改为2022
  4. localDateTime = localDateTime.with(ChronoField.YEAR, 2022);

日期计算。比如有些时候想知道这个月的最后一天是几号、下个周末是几号,通过提供的时间和日期API可以很快得到答案 。TemporalAdjusters提供的各种日期时间格式化的静态类,比如firstDayOfYear是当前日期所属年的第一天

  1. LocalDate localDate = LocalDate.now();
  2. LocalDate localDate1 = localDate.with(TemporalAdjusters.firstDayOfYear());

格式化时间。DateTimeFormatter默认提供了多种格式化方式,如果默认提供的不能满足要求,可以通过DateTimeFormatter的ofPattern方法创建自定义格式化方式

  1. LocalDate localDate = LocalDate.of(2019910);
  2. String s1 = localDate.format(DateTimeFormatter.BASIC_ISO_DATE);
  3. String s2 = localDate.format(DateTimeFormatter.ISO_LOCAL_DATE);
  4. //自定义格式化
  5. DateTimeFormatter dateTimeFormatter =   DateTimeFormatter.ofPattern("dd/MM/yyyy");
  6. String s3 = localDate.format(dateTimeFormatter);

解析时间。和SimpleDateFormat相比,DateTimeFormatter是线程安全的

  1. LocalDate localDate1 = LocalDate.parse("20190910", DateTimeFormatter.BASIC_ISO_DATE);
  2. LocalDate localDate2 = LocalDate.parse("2019-09-10", DateTimeFormatter.ISO_LOCAL_DATE);

Date与LocalDateTime转换。

  1. LocalDate localDate1 = LocalDate.parse("20190910", DateTimeFormatter.BASIC_ISO_DATE);
  2. LocalDate localDate2 = LocalDate.p/**
  3.  * LocalDateTime转毫秒时间戳
  4.  * @param localDateTime LocalDateTime
  5.  * @return 时间戳
  6.  */
  7. public static Long localDateTimeToTimestamp(LocalDateTime localDateTime)
  8.     {
  9.         try
  10.         {
  11.             ZoneId zoneId = ZoneId.systemDefault();
  12.             Instant instant = localDateTime.atZone(zoneId).toInstant();
  13.             return instant.toEpochMilli();
  14.         }
  15.         catch(Exception e)
  16.         {
  17.             e.printStackTrace();
  18.         }
  19.         return null;
  20.     }
  21.     /**
  22.      * 时间戳转LocalDateTime
  23.      * @param timestamp 时间戳
  24.      * @return LocalDateTime
  25.      */
  26. public static LocalDateTime timestampToLocalDateTime(long timestamp)
  27.     {
  28.         try
  29.         {
  30.             Instant instant = Instant.ofEpochMilli(timestamp);
  31.             ZoneId zone = ZoneId.systemDefault();
  32.             return LocalDateTime.ofInstant(instant, zone);
  33.         }
  34.         catch(Exception e)
  35.         {
  36.             e.printStackTrace();
  37.         }
  38.         return null;
  39.     }
  40.     /**
  41.      * Date转LocalDateTime
  42.      * @param date Date
  43.      * @return LocalDateTime
  44.      */
  45. public static LocalDateTime dateToLocalDateTime(Date date)
  46.     {
  47.         try
  48.         {
  49.             Instant instant = date.toInstant();
  50.             ZoneId zoneId = ZoneId.systemDefault();
  51.             return instant.atZone(zoneId).toLocalDateTime();
  52.         }
  53.         catch(Exception e)
  54.         {
  55.             e.printStackTrace();
  56.         }
  57.         return null;
  58.     }
  59.     /**
  60.      * LocalDateTime转Date
  61.      * @param localDateTime LocalDateTime
  62.      * @return Date
  63.      */
  64. public static Date localDateTimeToDate(LocalDateTime localDateTime)
  65. {
  66.     try
  67.     {
  68.         ZoneId zoneId = ZoneId.systemDefault();
  69.         ZonedDateTime zdt = localDateTime.atZone(zoneId);
  70.         return Date.from(zdt.toInstant());
  71.     }
  72.     catch(Exception e)
  73.     {
  74.         e.printStackTrace();
  75.     }
  76.     return null;
  77. }arse("2019-09-10", DateTimeFormatter.ISO_LOCAL_DATE);

SpringBoot中应用LocalDateTime

将LocalDateTime字段以时间戳的方式返回给前端 添加日期转化类

  1. public class LocalDateTimeConverter extends JsonSerializer < LocalDateTime >
  2. {
  3.     @Override
  4.     public void serialize(LocalDateTime value, JsonGenerator gen, SerializerProvider serializers) throws IOException
  5.     {
  6.         gen.writeNumber(value.toInstant(ZoneOffset.of("+8")).toEpochMilli());
  7.     }
  8. }

并在LocalDateTime字段上添加@JsonSerialize(using = LocalDateTimeConverter.class)注解,如下:

  1. @JsonSerialize(using = LocalDateTimeConverter.class)
  2. protected LocalDateTime gmtModified;

将LocalDateTime字段以指定格式化日期的方式返回给前端 在LocalDateTime字段上添加@JsonFormat(shape=JsonFormat.Shape.STRING, pattern="yyyy-MM-dd HH:mm:ss")注解即可,如下:

  1. @JsonFormat(shape=JsonFormat.Shape.STRING, pattern="yyyy-MM-dd HH:mm:ss")
  2. protected LocalDateTime gmtModified;

对前端传入的日期进行格式化 在LocalDateTime字段上添加@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")注解即可,如下:

  1. @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
  2. protected LocalDateTime gmtModified;

总结

LocalDateTime:Date有的我都有,Date没有的我也有。

继续阅读
历史上的今天
九月
9
weinxin
微信公众号
博客微信公众号,欢迎关注。
avatar
  • 版权声明 本文源自 闲人鹤 整理 发表于 2020年9月9日21:21:55
  • 除非特殊声明,本站文章均为原创,转载请务必保留本文链接。
稳定云服务器,香港VPS_美国VPS,高防服务器 硅云免费虚拟主机
匿名

发表评论

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

评论:2   其中:访客  1   博主  1
    • avatar 响石潭 3

      总结,你有你有,你啥都有,哈哈哈哈哈,date伤心

        • avatar 萧瑟

          @ 响石潭 学习时候看到了,不错,就转载了。作为自己的学习笔记。