为什么不要用Date表示时间

1. 引入


当我们写代码时或许会出现这种情况,但是这种代码在大佬眼里就是“屎山”!

大佬说:“我擦,这都什么年代了,怎么还在用Date来处理和表示时间!”

完了完了 . . .

2. 为啥Date遭嫌弃了

别的先不说,我们先来看几个关于Date用法的例子,这玩意真的好用吗?

2.1 打印此刻无可读性

Date rightNow = new Date();
System.out.println( "当前时刻:" + rightNow );
System.out.println( "当前年份:" + rightNow.getYear() );
System.out.println( "当前月份:" + rightNow.getMonth() );

输出结果为:
// 当前时刻:Fri Dec 13 21:46:34 CST 2019
// 当前年份:119
// 当前月份:11
  • 第一行:这打印结果你第一眼能看明白?可读性忒差了
  • 第二行:今天是2019年,你给我返回个119,没法读
  • 第三行:现在是12月份,你给我返回个11,这也没法读

2.2 构造一个指定年、月、日的时间

Date beforeDate = new Date(2019,12,12);


你看到啥了,连构造函数都被弃用了!

你可以再仔细瞅瞅,其实Date里的很多方法现在都已经弃用了!

都这样了,你项目还敢用这个吗?你醒醒吧!

3. LocalDateTime不香吗?

Java8开始,JDK中其实就增加了一系列表示日期和时间的新类,最典型的就是 LocalDateTime。直言不讳,这玩意的出现就是为了干掉之前JDK版本中的 Date老哥!

同样,我们也先来感受一下用法!

3.1 获取当前此刻的时间

LocalDateTime rightNow = LocalDateTime.now();
System.out.println( "当前时刻:" + rightNow );
System.out.println( "当前年份:" + rightNow.getYear() );
System.out.println( "当前月份:" + rightNow.getMonth() );
System.out.println( "当前日份:" + rightNow.getDayOfMonth() );
System.out.println( "当前时:" + rightNow.getHour() );
System.out.println( "当前分:" + rightNow.getMinute() );
System.out.println( "当前秒:" + rightNow.getSecond() );

// 输出结果:
当前时刻:2019-12-13T22:05:26.779
当前年份:2019
当前月份:DECEMBER
当前日份:13
当前时:22
当前分:5
当前秒:26

干得漂亮!

3.2 构造一个指定年、月、日的时间:

比如,想构造:2019年10月12月12日9时21分32秒

LocalDateTime beforeDate = LocalDateTime.of(2019, Month.DECEMBER, 12, 9, 21, 32);

没毛病!

3.3 修改日期

LocalDateTime rightNow = LocalDateTime.now();
rightNow = rightNow.minusYears( 2 ); // 减少 2 年
rightNow = rightNow.plusMonths( 3 ); // 增加 3 个月
rightNow = rightNow.withYear( 2008 ); // 直接修改年份到2008年
rightNow = rightNow.withHour( 13 ); // 直接修改小时到13时

够灵活!

3.4 格式化日期

LocalDateTime rightNow = LocalDateTime.now();
String result1 = rightNow.format( DateTimeFormatter.ISO_DATE );
String result2 = rightNow.format( DateTimeFormatter.BASIC_ISO_DATE );
String result3 = rightNow.format( DateTimeFormatter.ofPattern("yyyy/MM/dd") );
System.out.println("格式化后的日期(基本样式一举例):" + result1);
System.out.println("格式化后的日期(基本样式二举例):" + result2);
System.out.println("格式化后的日期(自定义样式举例):" + result3);

// 输出结果:
格式化后的日期(基本样式一举例)2019-12-13
格式化后的日期(基本样式二举例)20191213
格式化后的日期(自定义样式举例)2019/12/13

我无话可说,漂亮

3.5 时间反解析

给你一个陌生的字符串,你可以按照你需要的格式把时间给反解出来

LocalDateTime time = LocalDateTime.parse("2002--01--02 11:21",DateTimeFormatter.ofPattern("yyyy--MM--dd HH:mm"));
System.out.println("字符串反解析后的时间为:" + time);

// 输出:
字符串反解析后的时间为:2002-01-02T11:21

tql!

零零散散举了这么些例子,我想LocalDateTime怎么地也不输Date吧!

4. 线程安全性问题!

其实上面讲来讲去只讲了两者在用法上的差别,这其实倒还好,并不致命,可是接下来要讨论的线程安全性问题才是致命的!

其实以前我们惯用的· Date·时间类是可变类,这就意味着在多线程环境下对共享 Date变量进行操作时,必须由程序员自己来保证线程安全!否则极有可能翻车。

而自Java8开始推出的LocalDateTime却是线程安全的,开发人员不用再考虑并发问题,这点我们从LocalDateTime的官方源码中即可看出:

不说别的,就光一句:

This class is immutable and thread-safe.(不可变、线程安全!)

你就没有任何理由不用LocalDateTime

5. 日期格式化的选择

大家除了惯用 Date来表示时间之外,还有一个用于和Date连用的SimpleDateFormat 时间格式化类大家可能也戒不掉了!

SimpleDateFormat最主要的致命问题也是在于它本身并不线程安全,这在它的源码注释里已然告知过了:

那取而代之,我们现在改用什么呢?其实在前文已经用到啦,那就是了 DateTimeFormatter了,他也是线程安全的:

### SQL 中 `DATE` 关键字的用途与日期字面量的写法 在 SQL 中,`DATE` 关键字用于表示日期字面量或定义仅包含日期部分的数据类型。其主要用途是确保数据库系统将提供的字符串解析为日期值,并避免因隐式转换导致的错误或歧义。使用 `DATE` 关键字可以明确表达意图,确保数据库按照标准日期格式(`YYYY-MM-DD`)进行解析和存储[^1]。 例如,在更新语句中: ```sql UPDATE test_sal SET sal = 1000 WHERE empno = i.empno AND dt = DATE '2020-01-15'; ``` 这里的 `DATE '2020-01-15'` 明确指定了一个日期字面量,确保数据库系统将 `'2020-01-15'` 解析为 `DATE` 类型,而不是字符串或其他时间类型。 ### 日期字面量前使用 `DATE` 关键字的原因 在 SQL 中,字符串 `'2020-01-15'` 默认是 `CHAR` 或 `VARCHAR` 类型,而不是 `DATE` 类型。为了防止因类型不匹配而导致的错误,使用 `DATE` 关键字可以显式地将字符串转换为日期类型,确保与 `DATE` 类型的列进行正确比较或赋值。 此外,某些数据库系统(如 PostgreSQL 和 Oracle)要求使用 `DATE` 关键字来明确表示日期字面量,否则会抛出类型不匹配的错误。例如: ```sql SELECT * FROM access_log WHERE date = DATE '2023-10-01'; ``` 此查询确保数据库将 `'2023-10-01'` 明确解析为 `DATE` 类型,而不是字符串类型。 ### 日期字面量的标准格式 SQL 标准中定义的日期字面量格式为 `YYYY-MM-DD`,其中: - `YYYY` 表示年份(四位数) - `MM` 表示月份(01 到 12) - `DD` 表示日期(01 到 31) 例如: ```sql DATE '2020-01-15' DATE '1999-12-31' DATE '2023-03-01' ``` 这些格式确保数据库能够正确解析并存储日期值。 ### 不同数据库系统对 `DATE` 关键字的支持差异 不同数据库系统对 `DATE` 关键字的支持略有不同: - **MySQL**:通常不需要 `DATE` 关键字,可以直接使用字符串 `'2020-01-15'`,但使用 `DATE` 关键字也可以接受。 - **PostgreSQL**:要求使用 `DATE` 关键字来明确表示日期字面量。 - **Oracle**:支持 `DATE` 关键字,但通常使用 `TO_DATE()` 函数进行显式转换。 - **SQL Server**:不支持 `DATE` 关键字作为前缀,而是使用 `CONVERT()` 或 `CAST()` 函数进行类型转换。 ### 与其他日期时间类型的区别 SQL 中还支持 `DATETIME` 和 `TIMESTAMP` 类型,它们的格式为 `YYYY-MM-DD HH:MM:SS`,包含时间部分。相比之下,`DATE` 类型仅包含日期部分,不包含时间信息。因此,`DATE` 类型适用于仅需记录日期的场景,如生日、注册日期等[^1]。 ### 示例:使用 `DATE` 关键字进行查询和插入 #### 查询特定日期的记录 ```sql SELECT * FROM access_log WHERE date = DATE '2023-10-01'; ``` 该查询返回 `access_log` 表中日期为 `2023-10-01` 的所有记录。 #### 插入日期数据 ```sql INSERT INTO access_log (site_id, count, date) VALUES (1, 10, DATE '2023-10-01'); ``` 此语句向 `access_log` 表中插入一条包含日期 `2023-10-01` 的记录。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值