You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

584 lines
27 KiB
Markdown

2 years ago
# 16 | 用好Java 8的日期时间类少踩一些“老三样”的坑
你好,我是朱晔。今天,我来和你说说恼人的时间错乱问题。
在Java 8之前我们处理日期时间需求时使用Date、Calender和SimpleDateFormat来声明时间戳、使用日历处理日期和格式化解析日期时间。但是这些类的API的缺点比较明显比如可读性差、易用性差、使用起来冗余繁琐还有线程安全问题。
因此Java 8推出了新的日期时间类。每一个类功能明确清晰、类之间协作简单、API定义清晰不踩坑API功能强大无需借助外部工具类即可完成操作并且线程安全。
但是Java 8刚推出的时候诸如序列化、数据访问等类库都还不支持Java 8的日期时间类型需要在新老类中来回转换。比如在业务逻辑层使用LocalDateTime存入数据库或者返回前端的时候还要切换回Date。因此很多同学还是选择使用老的日期时间类。
现在几年时间过去了,几乎所有的类库都支持了新日期时间类型,使用起来也不会有来回切换等问题了。但,很多代码中因为还是用的遗留的日期时间类,因此出现了很多时间错乱的错误实践。比如,试图通过随意修改时区,使读取到的数据匹配当前时钟;再比如,试图直接对读取到的数据做加、减几个小时的操作,来“修正数据”。
今天,我就重点与你分析下时间错乱问题背后的原因,看看使用遗留的日期时间类,来处理日期时间初始化、格式化、解析、计算等可能会遇到的问题,以及如何使用新日期时间类来解决。
## 初始化日期时间
我们先从日期时间的初始化看起。如果要初始化一个2019年12月31日11点12分13秒这样的时间可以使用下面的两行代码吗
```
Date date = new Date(2019, 12, 31, 11, 12, 13);
System.out.println(date);
```
可以看到输出的时间是3029年1月31日11点12分13秒
```
Sat Jan 31 11:12:13 CST 3920
```
相信看到这里你会说这是新手才会犯的低级错误年应该是和1900的差值月应该是从0到11而不是从1到12。
```
Date date = new Date(2019 - 1900, 11, 31, 11, 12, 13);
```
你说的没错但更重要的问题是当有国际化需求时需要使用Calendar类来初始化时间。
使用Calendar改造之后初始化时年参数直接使用当前年即可不过月需要注意是从0到11。当然你也可以直接使用Calendar.DECEMBER来初始化月份更不容易犯错。为了说明时区的问题我分别使用当前时区和纽约时区初始化了两次相同的日期
```
Calendar calendar = Calendar.getInstance();
calendar.set(2019, 11, 31, 11, 12, 13);
System.out.println(calendar.getTime());
Calendar calendar2 = Calendar.getInstance(TimeZone.getTimeZone("America/New_York"));
calendar2.set(2019, Calendar.DECEMBER, 31, 11, 12, 13);
System.out.println(calendar2.getTime());
```
输出显示了两个时间,说明时区产生了作用。但,我们更习惯年/月/日 时:分:秒这样的日期时间格式,对现在输出的日期格式还不满意:
```
Tue Dec 31 11:12:13 CST 2019
Wed Jan 01 00:12:13 CST 2020
```
那,时区的问题是怎么回事,又怎么格式化需要输出的日期时间呢?接下来,我就与你逐一分析下这两个问题。
## “恼人”的时区问题
我们知道全球有24个时区同一个时刻不同时区比如中国上海和美国纽约的时间是不一样的。对于需要全球化的项目如果初始化时间时没有提供时区那就不是一个真正意义上的时间只能认为是我看到的当前时间的一个表示。
关于Date类我们要有两点认识
* 一是Date并无时区问题世界上任何一台计算机使用new Date()初始化得到的时间都一样。因为Date中保存的是UTC时间UTC是以原子钟为基础的统一时间不以太阳参照计时并无时区划分。
* 二是Date中保存的是一个时间戳代表的是从1970年1月1日0点Epoch时间到现在的毫秒数。尝试输出Date(0)
```
System.out.println(new Date(0));
System.out.println(TimeZone.getDefault().getID() + ":" + TimeZone.getDefault().getRawOffset()/3600000);
```
我得到的是1970年1月1日8点。因为我机器当前的时区是中国上海相比UTC时差+8小时
```
Thu Jan 01 08:00:00 CST 1970
Asia/Shanghai:8
```
对于国际化(世界各国的人都在使用)的项目,处理好时间和时区问题首先就是要正确保存日期时间。这里有两种保存方式:
* 方式一以UTC保存保存的时间没有时区属性是不涉及时区时间差问题的世界统一时间。我们通常说的时间戳或Java中的Date类就是用的这种方式这也是推荐的方式。
* 方式二,以字面量保存,比如年/月/日 时:分:秒一定要同时保存时区信息。只有有了时区信息我们才能知道这个字面量时间真正的时间点否则它只是一个给人看的时间表示只在当前时区有意义。Calendar是有时区概念的所以我们通过不同的时区初始化Calendar得到了不同的时间。
正确保存日期时间之后,就是正确展示,即我们要使用正确的时区,把时间点展示为符合当前时区的时间表示。到这里,我们就能理解为什么会有所谓的“时间错乱”问题了。接下来,我再通过实际案例分析一下,从字面量解析成时间和从时间格式化为字面量这两类问题。
**第一类是**对于同一个时间表示比如2020-01-02 22:00:00不同时区的人转换成Date会得到不同的时间时间戳
```
String stringDate = "2020-01-02 22:00:00";
SimpleDateFormat inputFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
//默认时区解析时间表示
Date date1 = inputFormat.parse(stringDate);
System.out.println(date1 + ":" + date1.getTime());
//纽约时区解析时间表示
inputFormat.setTimeZone(TimeZone.getTimeZone("America/New_York"));
Date date2 = inputFormat.parse(stringDate);
System.out.println(date2 + ":" + date2.getTime());
```
可以看到把2020-01-02 22:00:00这样的时间表示对于当前的上海时区和纽约时区转化为UTC时间戳是不同的时间
```
Thu Jan 02 22:00:00 CST 2020:1577973600000
Fri Jan 03 11:00:00 CST 2020:1578020400000
```
这正是UTC的意义并不是时间错乱。对于同一个本地时间的表示不同时区的人解析得到的UTC时间一定是不同的反过来不同的本地时间可能对应同一个UTC。
**第二类问题是**格式化后出现的错乱即同一个Date在不同的时区下格式化得到不同的时间表示。比如在我的当前时区和纽约时区格式化2020-01-02 22:00:00
```
String stringDate = "2020-01-02 22:00:00";
SimpleDateFormat inputFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
//同一Date
Date date = inputFormat.parse(stringDate);
//默认时区格式化输出:
System.out.println(new SimpleDateFormat("[yyyy-MM-dd HH:mm:ss Z]").format(date));
//纽约时区格式化输出
TimeZone.setDefault(TimeZone.getTimeZone("America/New_York"));
System.out.println(new SimpleDateFormat("[yyyy-MM-dd HH:mm:ss Z]").format(date));
```
输出如下我当前时区的Offset时差是+8小时对于-5小时的纽约晚上10点对应早上9点
```
[2020-01-02 22:00:00 +0800]
[2020-01-02 09:00:00 -0500]
```
因此有些时候数据库中相同的时间由于服务器的时区设置不同读取到的时间表示不同。这不是时间错乱正是时区发挥了作用因为UTC时间需要根据当前时区解析为正确的本地时间。
所以,**要正确处理时区,在于存进去和读出来两方面**存的时候需要使用正确的当前时区来保存这样UTC时间才会正确读的时候也只有正确设置本地时区才能把UTC时间转换为正确的当地时间。
Java 8推出了新的时间日期类ZoneId、ZoneOffset、LocalDateTime、ZonedDateTime和DateTimeFormatter处理时区问题更简单清晰。我们再用这些类配合一个完整的例子来理解一下时间的解析和展示
* 首先初始化上海、纽约和东京三个时区。我们可以使用ZoneId.of来初始化一个标准的时区也可以使用ZoneOffset.ofHours通过一个offset来初始化一个具有指定时间差的自定义时区。
* 对于日期时间表示LocalDateTime不带有时区属性所以命名为本地时区的日期时间而ZonedDateTime=LocalDateTime+ZoneId具有时区属性。因此LocalDateTime只能认为是一个时间表示ZonedDateTime才是一个有效的时间。在这里我们把2020-01-02 22:00:00这个时间表示使用东京时区来解析得到一个ZonedDateTime。
* 使用DateTimeFormatter格式化时间的时候可以直接通过withZone方法直接设置格式化使用的时区。最后分别以上海、纽约和东京三个时区来格式化这个时间输出
```
//一个时间表示
String stringDate = "2020-01-02 22:00:00";
//初始化三个时区
ZoneId timeZoneSH = ZoneId.of("Asia/Shanghai");
ZoneId timeZoneNY = ZoneId.of("America/New_York");
ZoneId timeZoneJST = ZoneOffset.ofHours(9);
//格式化器
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
ZonedDateTime date = ZonedDateTime.of(LocalDateTime.parse(stringDate, dateTimeFormatter), timeZoneJST);
//使用DateTimeFormatter格式化时间可以通过withZone方法直接设置格式化使用的时区
DateTimeFormatter outputFormat = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss Z");
System.out.println(timeZoneSH.getId() + outputFormat.withZone(timeZoneSH).format(date));
System.out.println(timeZoneNY.getId() + outputFormat.withZone(timeZoneNY).format(date));
System.out.println(timeZoneJST.getId() + outputFormat.withZone(timeZoneJST).format(date));
```
可以看到,相同的时区,经过解析存进去和读出来的时间表示是一样的(比如最后一行);而对于不同的时区,比如上海和纽约,最后输出的本地时间不同。+9小时时区的晚上10点对于上海是+8小时所以上海本地时间是晚上9点而对于纽约是-5小时差14小时所以是早上8点
```
Asia/Shanghai2020-01-02 21:00:00 +0800
America/New_York2020-01-02 08:00:00 -0500
+09:002020-01-02 22:00:00 +0900
```
到这里我来小结下。要正确处理国际化时间问题我推荐使用Java 8的日期时间类即使用ZonedDateTime保存时间然后使用设置了ZoneId的DateTimeFormatter配合ZonedDateTime进行时间格式化得到本地时间表示。这样的划分十分清晰、细化也不容易出错。
接下来我们继续看看对于日期时间的格式化和解析使用遗留的SimpleDateFormat会遇到哪些问题。
## 日期时间格式化和解析
每到年底就有很多开发同学踩时间格式化的坑比如“这明明是一个2019年的日期**怎么使用SimpleDateFormat格式化后就提前跨年了**”。我们来重现一下这个问题。
初始化一个Calendar设置日期时间为2019年12月29日使用大写的YYYY来初始化SimpleDateFormat
```
Locale.setDefault(Locale.SIMPLIFIED_CHINESE);
System.out.println("defaultLocale:" + Locale.getDefault());
Calendar calendar = Calendar.getInstance();
calendar.set(2019, Calendar.DECEMBER, 29,0,0,0);
SimpleDateFormat YYYY = new SimpleDateFormat("YYYY-MM-dd");
System.out.println("格式化: " + YYYY.format(calendar.getTime()));
System.out.println("weekYear:" + calendar.getWeekYear());
System.out.println("firstDayOfWeek:" + calendar.getFirstDayOfWeek());
System.out.println("minimalDaysInFirstWeek:" + calendar.getMinimalDaysInFirstWeek());
```
得到的输出却是2020年12月29日
```
defaultLocale:zh_CN
格式化: 2020-12-29
weekYear:2020
firstDayOfWeek:1
minimalDaysInFirstWeek:1
```
出现这个问题的原因在于这位同学混淆了SimpleDateFormat的各种格式化模式。JDK的[文档](https://docs.oracle.com/javase/8/docs/api/java/text/SimpleDateFormat.html)中有说明小写y是年而大写Y是week year也就是所在的周属于哪一年。
一年第一周的判断方式是从getFirstDayOfWeek()开始完整的7天并且包含那一年至少getMinimalDaysInFirstWeek()天。这个计算方式和区域相关对于当前zh\_CN区域来说2020年第一周的条件是从周日开始的完整7天2020年包含1天即可。显然2019年12月29日周日到2020年1月4日周六是2020年第一周得出的week year就是2020年。
如果把区域改为法国:
```
Locale.setDefault(Locale.FRANCE);
```
那么week yeay就还是2019年因为一周的第一天从周一开始算2020年的第一周是2019年12月30日周一开始29日还是属于去年
```
defaultLocale:fr_FR
格式化: 2019-12-29
weekYear:2019
firstDayOfWeek:2
minimalDaysInFirstWeek:4
```
这个案例告诉我们,没有特殊需求,针对年份的日期格式化,应该一律使用 “y” 而非 “Y”。
除了格式化表达式容易踩坑外SimpleDateFormat还有两个著名的坑。
第一个坑是,**定义的static的SimpleDateFormat可能会出现线程安全问题。**比如像这样使用一个100线程的线程池循环20次把时间格式化任务提交到线程池处理每个任务中又循环10次解析2020-01-01 11:12:13这样一个时间表示
```
ExecutorService threadPool = Executors.newFixedThreadPool(100);
for (int i = 0; i < 20; i++) {
//提交20个并发解析时间的任务到线程池模拟并发环境
threadPool.execute(() -> {
for (int j = 0; j < 10; j++) {
try {
System.out.println(simpleDateFormat.parse("2020-01-01 11:12:13"));
} catch (ParseException e) {
e.printStackTrace();
}
}
});
}
threadPool.shutdown();
threadPool.awaitTermination(1, TimeUnit.HOURS);
```
运行程序后大量报错且没有报错的输出结果也不正常比如2020年解析成了1212年
![](https://static001.geekbang.org/resource/image/3e/27/3ee2e923b3cf4e13722b7b0773de1b27.png)
SimpleDateFormat的作用是定义解析和格式化日期时间的模式。这看起来这是一次性的工作应该复用但它的解析和格式化操作是非线程安全的。我们来分析一下相关源码
* SimpleDateFormat继承了DateFormatDateFormat有一个字段Calendar
* SimpleDateFormat的parse方法调用CalendarBuilder的establish方法来构建Calendar
* establish方法内部先清空Calendar再构建Calendar整个操作没有加锁。
显然如果多线程池调用parse方法也就意味着多线程在并发操作一个Calendar可能会产生一个线程还没来得及处理Calendar就被另一个线程清空了的情况
```
public abstract class DateFormat extends Format {
protected Calendar calendar;
}
public class SimpleDateFormat extends DateFormat {
@Override
public Date parse(String text, ParsePosition pos)
{
CalendarBuilder calb = new CalendarBuilder();
parsedDate = calb.establish(calendar).getTime();
return parsedDate;
}
}
class CalendarBuilder {
Calendar establish(Calendar cal) {
...
cal.clear();//清空
for (int stamp = MINIMUM_USER_STAMP; stamp < nextStamp; stamp++) {
for (int index = 0; index <= maxFieldIndex; index++) {
if (field[index] == stamp) {
cal.set(index, field[MAX_FIELD + index]);//构建
break;
}
}
}
return cal;
}
}
```
format方法也类似你可以自己分析。因此只能在同一个线程复用SimpleDateFormat比较好的解决方式是通过ThreadLocal来存放SimpleDateFormat
```
private static ThreadLocal<SimpleDateFormat> threadSafeSimpleDateFormat = ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));
```
第二个坑是,**当需要解析的字符串和格式不匹配的时候SimpleDateFormat表现得很宽容**还是能得到结果。比如我们期望使用yyyyMM来解析20160901字符串
```
String dateString = "20160901";
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMM");
System.out.println("result:" + dateFormat.parse(dateString));
```
居然输出了2091年1月1日原因是把0901当成了月份相当于75年
```
result:Mon Jan 01 00:00:00 CST 2091
```
对于SimpleDateFormat的这三个坑我们使用Java 8中的DateTimeFormatter就可以避过去。首先使用DateTimeFormatterBuilder来定义格式化字符串不用去记忆使用大写的Y还是小写的Y大写的M还是小写的m
```
private static DateTimeFormatter dateTimeFormatter = new DateTimeFormatterBuilder()
.appendValue(ChronoField.YEAR) //年
.appendLiteral("/")
.appendValue(ChronoField.MONTH_OF_YEAR) //月
.appendLiteral("/")
.appendValue(ChronoField.DAY_OF_MONTH) //日
.appendLiteral(" ")
.appendValue(ChronoField.HOUR_OF_DAY) //时
.appendLiteral(":")
.appendValue(ChronoField.MINUTE_OF_HOUR) //分
.appendLiteral(":")
.appendValue(ChronoField.SECOND_OF_MINUTE) //秒
.appendLiteral(".")
.appendValue(ChronoField.MILLI_OF_SECOND) //毫秒
.toFormatter();
```
其次DateTimeFormatter是线程安全的可以定义为static使用最后DateTimeFormatter的解析比较严格需要解析的字符串和格式不匹配时会直接报错而不会把0901解析为月份。我们测试一下
```
//使用刚才定义的DateTimeFormatterBuilder构建的DateTimeFormatter来解析这个时间
LocalDateTime localDateTime = LocalDateTime.parse("2020/1/2 12:34:56.789", dateTimeFormatter);
//解析成功
System.out.println(localDateTime.format(dateTimeFormatter));
//使用yyyyMM格式解析20160901是否可以成功呢
String dt = "20160901";
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyyMM");
System.out.println("result:" + dateTimeFormatter.parse(dt));
```
输出日志如下:
```
2020/1/2 12:34:56.789
Exception in thread "main" java.time.format.DateTimeParseException: Text '20160901' could not be parsed at index 0
at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1949)
at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1777)
at org.geekbang.time.commonmistakes.datetime.dateformat.CommonMistakesApplication.better(CommonMistakesApplication.java:80)
at org.geekbang.time.commonmistakes.datetime.dateformat.CommonMistakesApplication.main(CommonMistakesApplication.java:41)
```
到这里我们可以发现使用Java 8中的DateTimeFormatter进行日期时间的格式化和解析显然更让人放心。那么对于日期时间的运算使用Java 8中的日期时间类会不会更简单呢
## 日期时间的计算
关于日期时间的计算我先和你说一个常踩的坑。有些同学喜欢直接使用时间戳进行时间计算比如希望得到当前时间之后30天的时间会这么写代码直接把new Date().getTime方法得到的时间戳加30天对应的毫秒数也就是30天\*1000毫秒\*3600秒\*24小时
```
Date today = new Date();
Date nextMonth = new Date(today.getTime() + 30 * 1000 * 60 * 60 * 24);
System.out.println(today);
System.out.println(nextMonth);
```
得到的日期居然比当前日期还要早根本不是晚30天的时间
```
Sat Feb 01 14:17:41 CST 2020
Sun Jan 12 21:14:54 CST 2020
```
出现这个问题,**其实是因为int发生了溢出**。修复方式就是把30改为30L让其成为一个long
```
Date today = new Date();
Date nextMonth = new Date(today.getTime() + 30L * 1000 * 60 * 60 * 24);
System.out.println(today);
System.out.println(nextMonth);
```
这样就可以得到正确结果了:
```
Sat Feb 01 14:17:41 CST 2020
Mon Mar 02 14:17:41 CST 2020
```
不难发现手动在时间戳上进行计算操作的方式非常容易出错。对于Java 8之前的代码我更建议使用Calendar
```
Calendar c = Calendar.getInstance();
c.setTime(new Date());
c.add(Calendar.DAY_OF_MONTH, 30);
System.out.println(c.getTime());
```
使用Java 8的日期时间类型可以直接进行各种计算更加简洁和方便
```
LocalDateTime localDateTime = LocalDateTime.now();
System.out.println(localDateTime.plusDays(30));
```
并且,**对日期时间做计算操作Java 8日期时间API会比Calendar功能强大很多**。
第一可以使用各种minus和plus方法直接对日期进行加减操作比如如下代码实现了减一天和加一天以及减一个月和加一个月
```
System.out.println("//测试操作日期");
System.out.println(LocalDate.now()
.minus(Period.ofDays(1))
.plus(1, ChronoUnit.DAYS)
.minusMonths(1)
.plus(Period.ofMonths(1)));
```
可以得到:
```
//测试操作日期
2020-02-01
```
第二还可以通过with方法进行快捷时间调节比如
* 使用TemporalAdjusters.firstDayOfMonth得到当前月的第一天
* 使用TemporalAdjusters.firstDayOfYear()得到当前年的第一天;
* 使用TemporalAdjusters.previous(DayOfWeek.SATURDAY)得到上一个周六;
* 使用TemporalAdjusters.lastInMonth(DayOfWeek.FRIDAY)得到本月最后一个周五。
```
System.out.println("//本月的第一天");
System.out.println(LocalDate.now().with(TemporalAdjusters.firstDayOfMonth()));
System.out.println("//今年的程序员日");
System.out.println(LocalDate.now().with(TemporalAdjusters.firstDayOfYear()).plusDays(255));
System.out.println("//今天之前的一个周六");
System.out.println(LocalDate.now().with(TemporalAdjusters.previous(DayOfWeek.SATURDAY)));
System.out.println("//本月最后一个工作日");
System.out.println(LocalDate.now().with(TemporalAdjusters.lastInMonth(DayOfWeek.FRIDAY)));
```
输出如下:
```
//本月的第一天
2020-02-01
//今年的程序员日
2020-09-12
//今天之前的一个周六
2020-01-25
//本月最后一个工作日
2020-02-28
```
第三可以直接使用lambda表达式进行自定义的时间调整。比如为当前时间增加100天以内的随机天数
```
System.out.println(LocalDate.now().with(temporal -> temporal.plus(ThreadLocalRandom.current().nextInt(100), ChronoUnit.DAYS)));
```
得到:
```
2020-03-15
```
除了计算外,还可以判断日期是否符合某个条件。比如,自定义函数,判断指定日期是否是家庭成员的生日:
```
public static Boolean isFamilyBirthday(TemporalAccessor date) {
int month = date.get(MONTH_OF_YEAR);
int day = date.get(DAY_OF_MONTH);
if (month == Month.FEBRUARY.getValue() && day == 17)
return Boolean.TRUE;
if (month == Month.SEPTEMBER.getValue() && day == 21)
return Boolean.TRUE;
if (month == Month.MAY.getValue() && day == 22)
return Boolean.TRUE;
return Boolean.FALSE;
}
```
然后使用query方法查询是否匹配条件
```
System.out.println("//查询是否是今天要举办生日");
System.out.println(LocalDate.now().query(CommonMistakesApplication::isFamilyBirthday));
```
使用Java 8操作和计算日期时间虽然方便但计算两个日期差时可能会踩坑**Java 8中有一个专门的类Period定义了日期间隔通过Period.between得到了两个LocalDate的差返回的是两个日期差几年零几月零几天。如果希望得知两个日期之间差几天直接调用Period的getDays()方法得到的只是最后的“零几天”,而不是算总的间隔天数**。
比如计算2019年12月12日和2019年10月1日的日期间隔很明显日期差是2个月零11天但获取getDays方法得到的结果只是11天而不是72天
```
System.out.println("//计算日期差");
LocalDate today = LocalDate.of(2019, 12, 12);
LocalDate specifyDate = LocalDate.of(2019, 10, 1);
System.out.println(Period.between(specifyDate, today).getDays());
System.out.println(Period.between(specifyDate, today));
System.out.println(ChronoUnit.DAYS.between(specifyDate, today));
```
可以使用ChronoUnit.DAYS.between解决这个问题
```
//计算日期差
11
P2M11D
72
```
从日期时间的时区到格式化再到计算你是不是体会到Java 8日期时间类的强大了呢
## 重点回顾
今天我和你一起看了日期时间的初始化、时区、格式化、解析和计算的问题。我们看到使用Java 8中的日期时间包Java.time的类进行各种操作会比使用遗留的Date、Calender和SimpleDateFormat更简单、清晰功能也更丰富、坑也比较少。
如果有条件的话我还是建议全面改为使用Java 8的日期时间类型。我把Java 8前后的日期时间类型汇总到了一张思维导图上图中箭头代表的是新老类型在概念上等价的类型
![](https://static001.geekbang.org/resource/image/22/33/225d00087f500dbdf5e666e58ead1433.png)
这里有个误区是认为java.util.Date类似于新API中的LocalDateTime。其实不是虽然它们都没有时区概念但java.util.Date类是因为使用UTC表示所以没有时区概念其本质是时间戳而LocalDateTime严格上可以认为是一个日期时间的表示而不是一个时间点。
因此在把Date转换为LocalDateTime的时候需要通过Date的toInstant方法得到一个UTC时间戳进行转换并需要提供当前的时区这样才能把UTC时间转换为本地日期时间的表示。反过来把LocalDateTime的时间表示转换为Date时也需要提供时区用于指定是哪个时区的时间表示也就是先通过atZone方法把LocalDateTime转换为ZonedDateTime然后才能获得UTC时间戳
```
Date in = new Date();
LocalDateTime ldt = LocalDateTime.ofInstant(in.toInstant(), ZoneId.systemDefault());
Date out = Date.from(ldt.atZone(ZoneId.systemDefault()).toInstant());
```
很多同学说使用新API很麻烦还需要考虑时区的概念一点都不简洁。但我通过这篇文章要和你说的是并不是因为API需要设计得这么繁琐而是UTC时间要变为当地时间必须考虑时区。
今天用到的代码我都放在了GitHub上你可以点击[这个链接](https://github.com/JosephZhu1983/java-common-mistakes)查看。
## 思考与讨论
1. 我今天多次强调Date是一个时间戳是UTC时间、没有时区概念为什么调用其toString方法会输出类似CST之类的时区字样呢
2. 日期时间数据始终要保存到数据库中MySQL中有两种数据类型datetime和timestamp可以用来保存日期时间。你能说说它们的区别吗它们是否包含时区信息呢
对于日期和时间,你还遇到过什么坑吗?我是朱晔,欢迎在评论区与我留言分享你的想法,也欢迎你把今天的内容分享给你的朋友或同事,一起交流。