一文详解Java LocalDateTime 一文详解卓依婷现状
xsobi 2024-12-17 17:07 2 浏览
逃离“时光旋涡” —— 从一次Date对象的“时区迷航”说起
想象一下,你的项目正紧锣密鼓地推进,突然间,一个看似不起眼的日期处理任务,却引发了一场“完美风暴”。一个简单的任务需求:存储用户生日并按其所在时区显示。使用传统的java.util.Date,你以为只需寥寥数行代码即可轻松搞定,却不料踏入了一个布满陷阱的雷区。时区转换的微妙差异,导致存储的生日莫名其妙地“漂移”了一天;线程并发下的不稳定性,又让日期数据的准确性如履薄冰。这不仅仅是一个技术问题,更成为了影响用户体验的“阿喀琉斯之踵”。
Java 8引入了全新的日期时间API (java.time包),彻底革新了日期和时间的处理方式,其中LocalDateTime、LocalDate、LocalTime三大类更是成为开发者手中的得力工具。遗憾的是,在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中,除了全能的LocalDateTime,java.time包还为我们带来了两位专注于单一维度的勇士——LocalDate和LocalTime。它们分别负责处理无时间的日期和无日期的时间信息,以其简洁而强大的设计,解决了众多日期时间处理中的痛点。接下来,让我们并肩探索LocalDate和LocalTime的奥秘,了解它们如何在各自的领域内大放异彩。
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类的主要优势:
明确的职责划分
- LocalDate和LocalTime分别专注于日期和时间的处理,这种分离使得模型更加清晰,避免了在单一对象中混合日期和时间信息可能导致的混淆。
- 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
相关推荐
- js向对象中添加元素(对象,数组) js对象里面添加元素
-
一、添加一个元素对象名["属性名"]=值(值:可以是一个值,可以是一个对象,也可以是一个数组)这样添加进去的元素,就是一个值或对象或数组...
- JS小技巧,如何去重对象数组?(一)
-
大家好,关于数组对象去重的业务场景,想必大家都遇到过类似的需求吧,这对这样的需求你是怎么做的呢。下面我就先和大家分享下如果是基于对象的1个属性是怎么去重实现的。方法一:使用.filter()和....
- 「C/C++」之数组、vector对象和array对象的比较
-
数组学习过C语言的,对数组应该都不会陌生,于是这里就不再对数组进行展开介绍。模板类vector模板类vector类似于string,也是一种动态数组。能够在运行阶段设置vector对象的长度,可以在末...
- 如何用sessionStorage保存对象和数组
-
背景:在工作中,我将[{},{}]对象数组形式,存储到sessionStorage,然后ta变成了我看不懂的形式,然后我想取之用之,发现不可能了~记录这次深刻的教训。$clickCouponIndex...
- JavaScript Array 对象 javascript的array对象
-
Array对象Array对象用于在变量中存储多个值:varcars=["Saab","Volvo","BMW"];第一个数组元素的索引值为0,第二个索引值为1,以此类推。更多有...
- JavaScript中的数组Array(对象) js array数组
-
1:数组Array:-数组也是一个对象-数组也是用来存储数据的-和object不同,数组中可以存储一组有序的数据,-数组中存储的数据我们称其为元素(element)-数组中的每一个元素都有一...
- 数组和对象方法&数组去重 数组去重的5种方法前端
-
列举一下JavaScript数组和对象有哪些原生方法?数组:arr.concat(arr1,arr2,arrn);--合并两个或多个数组。此方法不会修改原有数组,而是返回一个新数组...
- C++ 类如何定义对象数组?初始化数组?linux C++第43讲
-
对象数组学过C语言的读者对数组的概念应该很熟悉了。数组的元素可以是int类型的变量,例如int...
- ElasticSearch第六篇:复合数据类型-数组,对象
-
在ElasticSearch中,使用JSON结构来存储数据,一个Key/Value对是JSON的一个字段,而Value可以是基础数据类型,也可以是数组,文档(也叫对象),或文档数组,因此,每个JSON...
- 第58条:区分数组对象和类数组对象
-
示例设想有两个不同类的API。第一个是位向量:有序的位集合varbits=newBitVector;bits.enable(4);bits.enable([1,3,8,17]);b...
- 八皇后问题解法(Common Lisp实现)
-
如何才能在一张国际象棋的棋盘上摆上八个皇后而不致使她们互相威胁呢?这个著名的问题可以方便地通过一种树搜索方法来解决。首先,我们需要写一个函数来判断棋盘上的两个皇后是否互相威协。在国际象棋中,皇后可以沿...
- visual lisp修改颜色的模板函数 怎么更改visual studio的配色
-
(defunBF-yansemokuai(tuyuanyanse/ss)...
- 用中望CAD加载LISP程序技巧 中望cad2015怎么加载燕秀
-
1、首先请加载lisp程序,加载方法如下:在菜单栏选择工具——加载应用程序——添加,选择lisp程序然后加载,然后选择添加到启动组。2、然后是添加自定义栏以及图标,方法如下(以...
- 图的深度优先搜索和广度优先搜索(Common Lisp实现)
-
为了便于描述,本文中的图指的是下图所示的无向图。搜索指:搜索从S到F的一条路径。若存在,则以表的形式返回路径;若不存在,则返回nil。...
- 两个有助于理解Common Lisp宏的例子
-
在Lisp中,函数和数据具有相同的形式。这是Lisp语言的一个重大特色。一个Lisp函数可以分析另一个Lisp函数;甚至可以和另一个Lisp函数组成一个整体,并加以利用。Lisp的宏,是实现上述特色的...
- 一周热门
- 最近发表
- 标签列表
-
- grid 设置 (58)
- 移位运算 (48)
- not specified (45)
- patch补丁 (31)
- strcat (25)
- 导航栏 (58)
- context xml (46)
- scroll (43)
- element style (30)
- dedecms模版 (53)
- vs打不开 (29)
- nmap (30)
- webgl开发 (24)
- parse (24)
- c 视频教程下载 (33)
- android 开发环境 (24)
- paddleocr (28)
- listview排序 (33)
- firebug 使用 (31)
- transactionmanager (30)
- characterencodingfilter (33)
- getmonth (34)
- commandtimeout (30)
- hibernate教程 (31)
- label换行 (33)