时间问题在日常编程中会经常遇到,很多同学都会这块内容不是特别熟悉。我在这里做一些小的总结,梳理下我们会遇到的一些概念和问题,帮助大家解决时间问题的疑惑。
时区的概念
在初中或者高中地理课上,我们都学过时区的概念。地球上360度,被划分为24个时区,每一个时区的范围是15度。并且知道在中国,我们使用 东+8区 作为我们的本地时间的时区。
举个例子,18年暑假的时候,我去了欧洲,在那边基本每两天都会换一个国家或者地区。每次和爹妈打电话,我妈都会问下我几点了。我说是下午3点(15点),但是我妈都说,他们那边都已经晚上10(22点)了。可以看到,在同一个时刻,我和爹妈看到的“时间是不一样的”。难道真的是不一样的?其实我和爹妈所对话期间所处于的时刻是一样的,只是我们看到的“时间”是不一样的,这个时间被称做本地时间。也就是我看到的是 欧洲某个地区的15点,我妈看到的是北京时间的22点。
我相信,我们从很小的时候就接触了时区的概念。想想,你是否听过广播,看过新闻联播。每次这些节目开头,都会有一个句“北京时间 xx 点”,其实这里的北京时间就点明了时区。
通过上面的总结,我们是可以理解2021-1-20 08:00:00 +8:00
和 2021-1-20 00:00:00 +0:00
表示的是同一个时间/时刻。
计算机中如何存储时间
通过上节的内容,我们知道在不同的地区,大家使用的时区是不同的。例如在中国使用 GMT+8,在其他国家就是其他的形势了。想想如果我们在数据库中想存储一条时间数据,我们会选择使用这种本地时间吗?如果选择了本地时间,那么当另一个使用其他时间的人查询数据的时候,我们就需要对其进行一个转化。
先看几个概念:
UTC与GMT:通常情况下,我们可以认为GMT和UTC是一样的,都是用秒来计算的。
UTC与本地时:需要UTC 加或者减 时区差。
UTC与unix时间戳:unix时间戳其实就是我们在计算中常常说的 timestamp ,这个值在地球上的不同地方都是相同的(同一时刻),都是使用UTC时间的(1970-01-01 00:00:00)开始的算经过的秒数。
了解以上几个概念以后,我们就可以知道,计算机在通信或者存储的时候,可以选择本地时间+时区
或者直接使用 unix timestamp的方式进行存储(存储一个long值)。所以在数据库,或者平常使用中,我们都会直接使用long来对时间进行存储,在数据读取或者展示的时候选择一个timezone(时区),将对应的时间戳(timestamp)转化为对应时区的格式即可。例如 timesmap=0,如果在UTC时区展示就使用1970-01-01 00:00:00 +0:00
如果在UTC+8时间展示就使用1970-01-01 08:00:00
。以上的这种方式就是经典的**存储与展示分离的方式。
时间的展示问题
- 时区问题(timezone):同一个时间戳,不同的时区选择,表示出来的“样子”是不同的。
- 格式问题(locale):不同国家地区,展示同一个本地时间的“样子”也是不同的。
上面已经讲述过时区的问题的,在时间展示上除了时区以外还有格式问题。格式问题,其实就是不同国家或者地区对时间的展示是不同的。
例如同一个时区,同一个时间戳,使用不同的国际格式展示是不同的。
中国格式:2017-10-12 10:29:44
英国格式:Oct 12, 2017 10:29:44 AM