为什么计算机起始时间是1970年1月1日?

程序IT圈

共 2141字,需浏览 5分钟

 ·

2020-05-25 23:23



92d4ebf42c0a688165eca79a9cc4e48c.webp

d09c9a842f0d138827754c1b3fd71022.webp

3caaf57afcbeaa32ab1d117208671e6e.webp

578457fdd6d30387ca7b1f5fba71a532.webp

fff5ce1e6e67aa41195157a813027837.webp

443855299d532761f5fc2d76af10d191.webp

25a454754c7c61ba4ae66965f8eab7a6.webp

ca5311d45e62afd90b8cb958d8d9051b.webp


问题复现

1970-01-01对于开发者来说都是不陌生的,有些系统对于时间的处理如果不够好的话,就可能把时间显示成1970-01-01,所以经常有用户看到1970-01-01这个时间。

这种情况其实是很简单就可以复现的,当我们尝试使用以下Java代码定义时间:

Date date = new Date(0);
System.out.println(date);

打印出来的结果:

Thu Jan 01 08:00:00 CST 1970

通过Date的构造函数的Java Doc说明我们也能得到一些蛛丝马迹:

3fe47e3f97fbfdac6735e761167b6647.webp

该构造函数接收用户指定一个毫秒数,如new Date(1000),表示获得一个距离"epoch"有1000毫秒的时间。在Java中,这个时间是1970, 00:00:00 GMT。

0df730d5a76c9d0607842b48d8c277ea.webp

349a762561febd9bfdcb4a3925ecf108.webp

6a7da322a7816a1ae7fb41637c76dfbd.webp


时间戳

一切要从Unix操作系统诞生开始说起。

1969年8月,贝尔实验室的程序员肯汤普逊利用妻儿离开一个月的机会,开始着手创造一个全新的革命性的操作系统,他使用B编译语言在老旧的PDP-7机器上开发出了Unix的一个版本。

随后,汤普逊和同事丹尼斯里奇改进了B语言,开发出了C语言,重写了Unix,新版于1971年发布。

6a7739418e142976e5cb30ef7b2a728a.webp

在Unix被发明出来之后,需要在Unix上表示时间,就需要想办法定义一个能表示一份数据在某个特定时间之前已经存在的、完整的、可验证的数据来表示时间。

于是,Unix时间戳被定义出来,即通过当前时间和一个"纪元时间"进行对比,其间相差的秒数作为时间戳。

为了让Unix时间戳表示时间这种方式用的尽可能久,最初就把Unix诞生的时间1971-1-1定义成"纪元时间"。

104af63c6977dd1639c7c254c3180849.webp

18e6d2d8e2bc9993ab314d59ce1143f7.webp

993ba849d10e90c492298dc3642caf92.webp

6e8bbe1662d90c683dd18ae49b5ab9dc.webp


时间戳修改

除了开始时间是1971-1-1而不是1970-1-1外,最初的时间戳也不是每增加1秒时间戳就变动一次,而是每1/60秒都会改变一次时间戳。

另外,Unix是在1971年发明出来的,当时的计算机系统是32位,如果用32表示有整数,那么最大值是2147483647(2^31-1)。

那么,简单做一个数学计算,如果用当时的时间戳计算方式来表示时间的话,Unix时间戳最多可以使用4294967296/(60*60*24)/60 = 828.5天(一天有60*60*24秒,每1/60秒会占用一个时间戳)。

想象一下,设计出一个计算机系统,他的时间只能表示 828.5天,是不是很难让人接受,但是最初的Unix确实是这样的。

后来,Unix的开发者们也渐渐意识到这样不是长久之计,于是开始做出改变。

最开始,他们将每1/60秒改变一次时间戳修改成每1秒改变一次时间戳。这样时间戳可以表示的时间就又放大了60倍。这时候有828.5*60/365 = 136年。

这时候,一方面136年已经足够久了,纪元时间稍微向前调一下影响也不大。另外一方面为了方便记忆和使用。

于是就把纪元时间从1971-01-01调整到1970-01-01了。

7b8199b533405a1b49cb1e73ac3b14d6.webp

于是,随着后面各种开发语言的诞生,就都沿袭了1970-1-1这个设定。

所以,通常我们说的时间戳,就是指格林威治时间(GMT)1970年01月01日00时00分00秒起至现在的总秒数。

17788db97e20da8719bfe9e765e75356.webp

b4c90373375e9def2ebf3916ea86e456.webp

bc70f28b1e7a8be75328ce026e14d4af.webp

126d568dc4507d6aa224735512db8e81.webp

2d96b04b8598326c76bf5246b7699e0b.webp

b04f7a256854a3f2fbff1628f97da5a1.webp


纪元时间与时区

我们前面所提到的纪元时间的设置,都是基于格林威治标准时间的,即GMT时间。

但是世界上各个地区有自己的时区,都需要基于GMT时间进行调整。

1970-01-01 08:00:00的显示显然是受到了时区的影响,因为中国处于东八区,所以时间会比标准时间早8小时,而标准时间应该是1970-01-01 00:00:00。

应该很多人都记得《苹果"1970 事件"》,在几年前,一个名为vista980622的网友在国外网站Reddit的论坛上发表了一篇“把iPhone时间改成1970年1月1日,手机即可永远变砖”的帖子。

629a536da279c000daa6b3ff561c7e16.webp

在该帖子发布不久,很多人都不相信,抱着试试看的态度将手机的时间设置成1970年1月1日,结果手机关机后重新开机真的变砖了。

因为我们处于东八区,时间比标准时间要快8小时,如果我们把时间调整成1970-01-01 00:00:00,那么标准时间就会是比这个时间少8小时,即1969年12月31日16时0分0秒。

但是,IOS设备是以UTC时区(GMT时间)的1970年1月1日0点0时0秒为界限,数值为0,用户把时间调整到1969年12月31日16时0分0秒,系统就要出现负值的时间。

系统版本为IOS 8.0至IOS 9.3 beta3,并且搭载64位处理器(即处理器为A7-A9X的设备)的苹果设备都会触发这个Bug,导致变砖!

2ee4c5df9e7cee15b1f7d6819f44b215.webp

2681d1dd68fd80da4d71a6dbf1c83077.webp

87508639cad670705724db9d4d58693e.webp

a110f5cc774408aa78c3808eeee66b2a.webp



参考资料:

http://en.wikipedia.org/wiki/Unix_time#History 

http://mobile.zol.com.cn/569/5691666_all.html


微信群

仅供关注公众号「程序IT圈」 的读者加入!进群技术交流与日常交流讨论,技术内容不限于Java、Python、算法学习、求职内推、资料分享等等 。当然日常吹水也是可以的,但是禁止随意开车 。严禁一切推广链接、二维码,一经发现送上飞机票?。

加群方式:扫描二维码,备注【加群】,微信号:it4201 。

同意后,手动拉你进群~


好文章,我在看❤️

浏览 27
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报