MongoDB 常见问题解答:时间与时区
这期主要来聊一下 MongoDB 中的时区问题。
这个问题虽然简单,总有同学会问到,集中解答一下。
提出这个问题通常是因为在使用 Mongo Shell 后发现其中展示的时间比中国时间落后 8 小时而产生的。
比如:插入时间是 2021 年 1 月 1 日 00:00:00,在 Shell 中看到的则是 ISODate("2020-12-31T16:00:00Z"),很多人由此产生疑惑,8 小时去哪了?是不是出错了?
理解这个问题首先要理解时区的概念。当你同时向一个身处中国的人和身处美国的人提问现在是几点时,中国人回答中午 12 点,美国人回答的却是凌晨 4 点。这有区别吗?他们说错了吗?他们都没错。他们回答不一样是因为身处不同的时区,但是他们指代的都是同一个时间:现在。
也可以说,全世界只有一个时间,只是大家描述它的方式不一样——根据自己身处的地区。而有个现状会把这个问题搞得更复杂,那就是夏令时。
我国现在已经不实施夏令时了(对,过去有过!暴露年龄!),但是世界上有很多国家仍然保留夏令时的习惯。这让原本复杂的时间表达更加雪上加霜。这在实施同样的时间规则的人之间不会造成问题。
但是在国际化的今天,你知道一个中国人跟一个实施了夏令时的德国人约一个会议时间有多难?或者在看到一个德国时间 2020 年 5 月 7 日 18 点,你知道它是英国时间的几点吗?你不光要知道英德的时差,还要知道 2020 年 5 月 7 日这个时间在德国是不是在夏令时的影响范围内,以及在英国它是不是在夏令时的影响范围内。现在有没有觉得小瞧了时间的复杂性?
为了简化这些复杂的问题,有了 UTC (Universal Time Coordinated)时间标准以及表达时间的标准 ISO 8601,也就是我们在 MongoDB 中看到的 ISODate。
ISODate 的表示方法简单来说就是年月日时分秒+时区。以我们前面的例子来说,身处中国时我们的时区是 UTC+8,用 ISODate 表达则是 ISODate("2021-01-01T00:00:00+0800")。这个时间在 MongoDB 中的表达是 ISODate("2020-12-31T16:00:00Z"),其中的 Z 表示 UTC(或者 UTC+0)。由于时区的不同导致了表达方式不同,但这两个时间指代的是同一时刻,所以两者没有什么不一样。而在展示时,则应该根据你的用户所处的时区来决定到底显示哪一个结果。
需要注意的是:有些语言的驱动(例如 Java/C#)在读取到时间时会自动转换为服务器时间,所以不需要再人为转换。有些语言(例如 Python)则不会自动转换。使用时应该根据你的实际情况决定处理方式。
作者介绍:张耀星
MongoDB 大中华区首席咨询顾问,供职于 MongoDB 售后服务团队 5 年+,拥有近 10 年 MongoDB 使用经验。
评论