关于时间,每个程序员都应了解的事

关于时间,每个程序员都应了解的事

注: 英文原文由 @程序员的那些事 在微博推荐,感谢 @泉州一中-刘家昌  编译完成初稿。在译文基础上, 伯乐 在线根据维基百科等资料有补充和改动。 (如需转载,请保留本段说明。)

一些关于时间的注解:

●UTC / 协调世界时: 又称“世界标准时间”或“世界协调时间”,简称 UTC (从英文“ C oordinated  U niversal  T ime”/法文“ T emps  U niversel  C ordonné”而来),是最主要的世界时间标准,其以原子时秒长为基础,在时刻上尽量接近于格林尼治平时。中国大陆称之为“协调世界时”。台湾称之为“世界統一時間”。

●GMT: 因为本初子午线被英国御用天文学家艾里武断地选定为位于英国格林尼治天文台的一条经线, UTC 又曾称为“格林尼治标准时间” (Greenwich Mean Time, GMT)。

 
设于英国 格林尼治皇家天文台 大门外的24小时制电子大钟,显示格林尼治标准时间。

 

●其他时区都可写为与 UTC 时间的偏移量。北京时间是 UTC+8(俗称“东八区”)。例如某日的 00:00 UTC 是同日的北京时间 08:00。

●夏令时  并不影响 UTC。它只是当地政府关于改变其时区(与 UTC 的偏移量)的决定。 例如,GMT 在冬季是英国的国家时区,而夏季则选用英国夏令时(BST)。

● 闰秒 :  根据国际惯例,通过在每个 UTC 年的十二月,或者六月的最后一秒,引入一秒闰秒,来使 UTC 与物理现实(UT1,天文时间量度)保持 0.9 秒以内的差距。(UTC 完全是人类武断的定义,而平均太阳日的长度正以非常缓慢的速度增加中)

●闰秒并不需提前六个月宣布。这对于时长超过六个月的精密时间计划是重大的问题。

● Unix 时间 : 亦或称 POSIX时间 ,以 Unix 纪元初(1970年1月1日0时0分0秒)至今的总秒数度量,不包括闰秒。 Unix 时间不受时区和夏令时的影响。

Year_2038_problem 2038年问题

2038年 1月19日 3时14分07秒,32位系统的UNIX时间将会被重置。

 

●根据 POSIX.1 标准 ,Unix 时间应该通过重复前一秒处理闰秒。例如:

59.00
59.25
59.50
59.75
59.00 ← 重复
59.25
59.50
59.75
00.00 ← 增加
00.25

这一方法有利有弊:你无法重现一个闰秒,而且时间可能会倒退;但从另一个角度来说,每天的时间却确切为 86,400 秒长,输出人类可读的时分秒之时,你并不需要列出一个过去和未来的所有闰秒表。

● ntpd  (Network Time Protocol daemon)在从上流服务器处接收到“闰秒位(leap bits)”后,应当完成这次时间重复。 但我也曾见过它违背这一标准:系统进入到未来的一秒,然后缓慢地偏移回正确的时间。

 

每个 程序员 都应了解的关于时间的事:

●时区是一个表示层的问题!   你多数的代码不应处理时区或本地时问题,请总使用 Unix 时间。

●使用 Unix 时间作为度量,它是 UTC 时间 —— 易于获取,而且没有时区、夏令时(和闰秒)。

●保存时间时使用 Unix 时间,它是一个整数。

●如果你希望保存人类可读的时间(例如在日志中),考虑将其   Unix 时间共同保存, 而不是 取代  Unix 时间。

●显示时间时,同时显示所在时区——一个未知时区偏移量的时间是无用的。

●系统时钟并不精确。

●你在一个网络中? 那么请注意所有系统的时钟都有不同的不精确性。

●系统时钟可以——或者会——因你所控制外的事情向后或向前跳跃。你的程序必须能够处理这一问题。

●每[实际]秒的[时钟]滴答并不精确且易变。多数情况它会随温度而变化。

●不要盲目地使用  gettimeofday() 。如果你需要一个单调递增的时钟,选择clock_gettime() .

● ntpd  可以通过以下两种方式改变系统时间:

+跳跃: 直接将系统时钟即时地向前或向后调整为正确时间。

+转换: 修改系统时钟的频率,使系统时间慢慢地偏移为正确时间。

转换这一方法破坏性较小,因而它较受欢迎。但只当修复小的时间误差时此方法才有效。

 

值得一提的:

●根据相对论,每个“观察者”的“一秒钟”都是不同的。远程时钟频率相对一个“观察者”来说,受到相对速度和重力的影响。GPS 卫星中的时间已根据相对论效应调整。

●MySQL(高于 4.x 和 5.x 的版本)将  DATETIME  类型保存为一个 “YYYY-MM-DD HH:MM:SS” 串的二进制编码。其本身并不保存时差,使用时根据 @@session.time_zone 指示时区解释。

mysql> insert into times values(now());
mysql> select unix_timestamp(t) from times;
1310128044
mysql> SET SESSION time_zone='+0:00';
mysql> select unix_timestamp(t) from times;
1310164044

如果你很介意这件事,那么使用  UNIX_TIMESTAMP()  和  FROM_UNIXTIME()  函数,将时间保存为数字。(UNIX_TIMESTAMP() 将参数时间串使用当前 time_zone 的设定解释,并转换为 Unix 时间数字,如果当地规定了夏令时,也将按夏令时解释)

还有  TIMESTAMP  类型,将时间保存为 Unix 时间。它有不同的 用法 。

 

英文原文: unix4lyfe ,编译: @泉州一中-刘家昌

译文链接: http://blog.jobbole.com/32534/

发表评论

电子邮件地址不会被公开。 必填项已用*标注