一文详解Java LocalDateTime 一文详解FC和FB的区别与用法

逃离“时光旋涡” —— 从一次Date对象的“时区迷航”说起

想象一下,你的项目正紧锣密鼓地推进,突然间,一个看似不起眼的日期处理任务,却引发了一场“完美风暴”。一个简单的任务需求:存储用户生日并按其所在时区显示。使用传统的java.util.Date,你以为只需寥寥数行代码即可轻松搞定,却不料踏入了一个布满陷阱的雷区。时区转换的微妙差异,导致存储的生日莫名其妙地“漂移”了一天;线程并发下的不稳定性,又让日期数据的准确性如履薄冰。这不仅仅是一个技术问题,更成为了影响用户体验的“阿喀琉斯之踵”。

Java 8引入了全新的日期时间API (java.time包),彻底革新了日期和时间的处理方式,其中LocalDateTimeLocalDateLocalTime三大类更是成为开发者手中的得力工具。遗憾的是,在java LTS版本已经发展到21的今天,Java 8引入的日期时间的类还未被大多数开发者熟练使用,本文会阐述这些又“新”又“旧”的类的基本用法,快来看看你都”学废“了吗?

LocalDateTime 时间与日期的完美融合

在Java 8中,LocalDateTime类作为java.time包的一员,标志着日期和时间处理的新篇章。它结合了LocalDate的日期信息和LocalTime的时间信息,提供了一个不含时区信息的日期时间表示方式,适用于那些只需要关注本地日期和时间,而不需要考虑时区差异的场景。

创建

LocalDateTime代表的是一个没有时区信息的日期和时间组合,适用于记录或显示本地日期和时间,比如会议安排、生日提醒等。创建LocalDateTime实例可以通过多种方式:

  • 当前日期时间:LocalDateTime now = LocalDateTime.now(); 获取当前系统默认时区的日期和时间。
  • 指定日期时间:LocalDateTime dt = LocalDateTime.of(2024, Month.JANUARY, 1, 12, 30); 创建特定的日期和时间。
  • 字符串解析:LocalDateTime parsed = LocalDateTime.parse("2024-01-01T12:30"); 从ISO 8601标准格式的字符串解析日期时间。

访问与修改

LocalDateTime提供了丰富的getter方法用于访问日期和时间的各个组成部分,如getYear(), getMonth(), getDayOfMonth(), getHour(), getMinute()等。同时,它还支持对日期时间进行调整:

  • 日期调整:如withDayOfMonth(15)将日期调整到当月的15号。
  • 时间调整:如plusHours(2)表示加上2小时。

操作与计算

LocalDateTime提供了加减日期或时间的方法,允许进行日期时间的计算:

  • 加减日期时间:LocalDateTime future = dt.plusDays(10); 计算10天后的日期时间。
  • 计算两个日期时间的差距:结合Duration或Period可以计算两个LocalDateTime之间的差异。

格式化与解析

为了适应不同的展示或存储需求,LocalDateTime支持自定义格式化和解析:

  • 格式化输出:DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String formatted = dt.format(formatter); 格式化为指定样式。
  • 解析字符串:LocalDateTime fromString = LocalDateTime.parse("2023-01-01 12:30", formatter); 将格式化的字符串解析回LocalDateTime。

与时间相关的转换

虽然LocalDateTime不包含时区信息,但可以与其他包含时区的类(如ZonedDateTime)进行转换,以便在全球范围内使用:

  • 结合时区:通过ZonedDateTime zdt = dt.atZone(ZoneId.systemDefault()); 将本地日期时间转换为特定时区的日期时间。
  • 去除时区信息:从ZonedDateTime转换回LocalDateTime,可以使用zdt.toLocalDateTime()。

注意事项

  • 时区意识:虽然LocalDateTime不携带时区信息,但在处理跨国或跨时区的数据时,应考虑使用ZonedDateTime以避免混淆。
  • 不可变性:如同LocalDate和LocalTime,LocalDateTime也是不可变的,所有修改操作都会返回一个新的实例。
  • 线程安全:由于不可变性,LocalDateTime是线程安全的,适合在并发环境下使用。

LocalDate、LocalTime:日期与时间的纯粹表达

在Java 8中,除了全能的LocalDateTimejava.time包还为我们带来了两位专注于单一维度的勇士——LocalDateLocalTime。它们分别负责处理无时间的日期和无日期的时间信息,以其简洁而强大的设计,解决了众多日期时间处理中的痛点。接下来,让我们并肩探索LocalDateLocalTime的奥秘,了解它们如何在各自的领域内大放异彩。

LocalDate:纯粹的日期

LocalDate代表不含时间信息的日期,专注于年月日的管理,非常适合处理生日、纪念日、合同截止日期等场景。

  • 创建与获取:通过LocalDate.now()获取当前日期,或者使用LocalDate.of(2023, Month.JANUARY, 1)指定日期。
  • 日期操作:轻松进行日期的加减,如LocalDate tomorrow = today.plusDays(1)获取明天的日期。
  • 比较与判断:提供isBefore(), isAfter(), isEqual()等方法,方便比较日期先后。
  • 格式化与解析:与DateTimeFormatter结合,实现日期的自定义格式化输出和解析。

LocalTime:时间的精准表达

LocalTime专注于处理一天中的时间,没有日期信息,适用于记录营业时间、事件发生的具体时间点。

  • 初始化与获取:使用LocalTime.now()获取当前时间,或LocalTime.of(14, 30)指定具体时间。
  • 时间调整:通过plusHours(), minusMinutes()等方法,实现对时间的加减操作。
  • 比较与计算:提供方法来比较时间的早晚,以及计算两个时间点的差距,使用Duration来表示时间差。
  • 格式化与解析:同样支持与DateTimeFormatter的配合,进行时间的格式化输出和解析。

相互转换与应用场景

  • 相互结合:LocalDateTime实际上就是LocalDate和LocalTime的组合体,通过LocalDateTime.of(date, time)或各自类的atTime()、atDate()方法可以互相转换。
  • 专注单一维度:在处理特定场景时,单独使用LocalDate或LocalTime可以避免不必要的复杂性,提高代码的可读性和维护性。
  • 时区无关性:两者都不包含时区信息,适合处理与特定时区无关的日期或时间信息。

注意事项

  • 时区意识:虽然LocalDate和LocalTime不包含时区信息,但在处理跨越时区的数据时,应考虑使用ZonedDateTime。
  • 不可变性与线程安全:与LocalDateTime类似,LocalDate和LocalTime也是不可变类,所有修改操作返回新实例,保证了线程安全。
  • 精确计算:进行日期时间的加减操作时,应考虑使用Period和Duration来精确表达时间跨度。

LocalDateTime、LocalDate、LocalTime vs java.util.Date

相较于传统的java.util.Date类,在设计哲学、功能特性以及易用性上实现了显著的飞跃。以下是它们相对于Date类的主要优势:

明确的职责划分

  • LocalDateLocalTime分别专注于日期和时间的处理,这种分离使得模型更加清晰,避免了在单一对象中混合日期和时间信息可能导致的混淆。
  • LocalDateTime虽然综合了日期和时间,但其设计初衷就是为了清晰地处理含有日期和时间信息的场景,相比Date类的多功能混杂,使用起来更为直观。

不可变性与线程安全

  • 所有java.time类,包括LocalDateTime、LocalDate、LocalTime,均为不可变对象。这意味着一旦创建,它们的值就不会改变,这有助于在并发环境下保持数据的一致性,而Date类则是可变的,容易在多线程环境下引起数据竞争问题。

丰富的API设计

  • 新API提供了更丰富且直观的操作方法,如日期时间的加减、比较、格式化等,避免了Date类中繁琐且容易出错的日期时间计算。
  • 例如,LocalDate.plusDays()直接增加了天数,而Date类需要通过Calendar类间接操作,过程更为复杂。

时区处理的明确性

  • 虽然LocalDateTime、LocalDate、LocalTime本身不包含时区信息,但这实际上是为了清晰地区分本地时间与全球时间的概念。对于需要时区处理的场景,Java 8提供了ZonedDateTime类,它比Date类的时区处理更为灵活和精确。
  • 相比之下,Date类虽然包含时区信息,但处理时区问题时往往显得笨拙且不够明确。

ISO 8601标准遵循

  • LocalDateTime、LocalDate、LocalTime在格式化和解析上遵循ISO 8601国际标准,如YYYY-MM-DD和HH:mm:ss,这使得日期时间的字符串表示更加通用和标准化,便于跨系统和国际化应用。
  • 而Date类的默认字符串表示并不遵循任何标准格式,需要手动格式化和解析,增加了复杂度。

性能与精确性

  • java.time包采用了更高效的设计,尤其是对于日期时间的计算和存储,相比Date类在性能上有一定的提升。
  • 此外,新API提供了纳秒级的精度,而Date类的精度仅为毫秒级,对于需要极高时间精度的应用来说,这是一个显著的进步。

总结

可以看到,这些JDK8引入的Localxxx类,都可以让我们对日期和时间的处理更加准确和方便,虽然继续使用java.util.Date也在多数场景可以满足需求,让我想起一张远古的图

不过在不同的场景使用最适合的类和方法,尽量把代码写的简洁和优雅,才是一个程序猿不断追求的目标吧。

看到这里了,点个赞再走呗

原文:https://juejin.cn/post/7394345043131858970

作者:podongfeng

#记录我的2024#

相关文章

大数据编程入门:Java日期及时间 java 大数据

在使用Java语言写一些小程序的时候,总少不了与时间打交道,那么在Java中我们需要输出日期时间时应该怎么编写呢?今天小编将为大家带来一篇大数据编程入门:Java日期时间的文章。一、Java Date...

JAVA常用时间工具类(LocalDateTime)

判断当前时间是在某个时间段内Date转换为LocalDateTimeLocalDateTime转换为Date获取指定日期的毫秒获取指定日期的秒获取指定时间的指定格式获取"yyyy-MM-dd...

100个Java工具类之53:日历类Calendar

Calendar类主要作用是提供日期转换。它是一个抽象类,需调用特定方法getInstance来获取实例。下文是对其通用用法的介绍。一、获取当前时间//获取当前时间 返回Date类型 Calendar...

Java培训 | JAVA日期类 java中的日期类型

我们在程序开发过程中经常遇到对日期进行处理的需求,此时能熟练掌握Java API中提供的日期相关类就显得非常重要,可以顺利帮助我们解决遇到的问题。然而在Java8以前,日期和时间处理一直被程序员抱怨太...

Java简单日期类型时间格式的详解 java 系统日期

名词解释时刻:时刻是时间上的某个瞬间,,或是时间轴上的某个点。时刻可以作为时间间隔的起点和终点时代:一个时代是用作测量其他时刻的起源的瞬间。时代是在1970-01-01T00:00:00Z。时代之后的...

java组件huTool日期DateUtil工具的使用

Hutool是一个Java工具包,也只是一个工具包,它帮助我们简化每一行代码,减少每一个方法。日期处理也是日常工作中常用到的,下面说一下工具下用到的一些方法。计算between(Date begin...