我希望将这个问题及其答案作为处理夏令时的最终指南,特别是处理实际转换时。

如果你有什么要补充的,请做

许多系统都依赖于保持准确的时间,问题是由于夏时制而导致的时间变化——向前或向后移动时钟。

例如,一个订单接受系统中的业务规则取决于订单的时间——如果时钟发生变化,规则可能就不那么清晰了。应如何保持订单的时间?当然,有无数的场景——这只是一个例证。

你是如何处理日光节约问题的?您的解决方案包含哪些假设?(在此处查找上下文)

同样重要,如果不是更重要的话:

你尝试了什么但没有奏效?为什么不起作用?

我会对编程、操作系统、数据持久性和其他相关方面感兴趣。

一般的答案很好,但我也希望看到细节,特别是如果它们只在一个平台上提供的话。


当前回答

以下是我的经验:-

(不需要任何第三方库)

在服务器端,以UTC格式存储时间,以便数据库中的所有日期/时间值都采用单一标准,而不考虑用户、服务器、时区或DST的位置。在UI层或发送给用户的电子邮件中,您需要根据用户显示时间。为此,您需要有用户的时区偏移量,以便将此偏移量添加到数据库的UTC值中,这将导致用户的本地时间。您可以在用户注册时获取他们的时区偏移,也可以在web和移动平台中自动检测他们。对于网站,JavaScript的函数getTimezoneOffset()方法是1.0版以来的标准,并且与所有浏览器兼容。(参考:http://www.w3schools.com/jsref/jsref_getTimezoneOffset.asp)

其他回答

实际上,kernel32.dll不导出SystemTimeToTzSpecificLocation。但是,它确实导出以下两个:SystemTimeToTzSpecificLocalTime和TzSpecificLocalTimeToSystemTime。。。

这是一个重要而令人惊讶的棘手问题。事实是,对于坚持时间没有完全令人满意的标准。例如,SQL标准和ISO格式(ISO 8601)显然不够。

从概念角度来看,通常处理两种类型的时间-日期数据,区分它们很方便(上述标准没有):“物理时间”和“民事时间”。

时间的“物理”瞬间是物理学处理的连续宇宙时间线中的一个点(当然忽略了相对论)。例如,如果可以忽略闰秒,则可以在UTC中对这个概念进行适当的编码。

“民用”时间是遵循民用规范的日期时间规范:这里的时间点完全由一组日期时间字段(Y、M、D、H、MM、S、FS)加上TZ(时区规范)(实际上也是“日历”;但假设我们将讨论限于公历)来指定。时区和日历共同允许(原则上)从一个表示映射到另一个表示。但是,民用和物理时间瞬间是本质上不同类型的量值,它们应该在概念上分开并以不同的方式对待(一个类比:字节和字符串的数组)。

这个问题令人困惑,因为我们可以互换地谈论这些类型的事件,并且因为公民时代会受到政治变化的影响。这个问题(以及区分这些概念的必要性)在未来的事件中变得更加明显。示例(摘自我在这里的讨论。

约翰在日历上记录了某个日期的事件提醒2019-Jul-27,10:30:00,TZ=智利/圣地亚哥,(已偏移GMT-4,因此它对应于UTC 2019-Jul-27 14:30:00)。但总有一天未来,该国决定将TZ偏移量改为GMT-5。

现在,当这一天到来。。。该提醒是否应在

A) 2019-Jul-27 10:30:00智利/圣地亚哥=UTC时间2019-Jur-27 15:30:00?

or

B) 2019-Jul-27 9:3:00智利/圣地亚哥=UTC时间2019-Jul-27 14:30:00?

没有正确的答案,除非你知道约翰在概念上的意思当他告诉日历“请在2019年7月27日10:30:00给我打电话TZ=智利/圣地亚哥”。

他是说“公民约会时间”吗10:30“)?在这种情况下,A)是正确答案。

或者他是指“物理上的瞬间”,连续波中的一个点我们宇宙的时间线,比如,“下一次日食在这种情况下,答案B)是正确的。

一些Date/Time API正确地实现了这一区别:其中的Jodatime是下一个(第三个!)Java DateTime API(JSR 310)的基础。

您需要了解Olson-tz数据库,可从ftp://elsie.nci.nih.gov/pub http://iana.org/time-zones/.它每年更新多次,以应对世界各地不同国家在冬季和夏季(标准时间和夏令时)之间切换的时间(以及是否切换)的最后时刻变化。2009年,最后一次发布是2009年;2010年为2010n;2011年为2011n;2012年5月底,发布时间为2012c。注意,在两个独立的存档(tzcode20xxy.tar.gz和tzdata20xxy.tar.gz)中有一组代码来管理数据和实际时区数据本身。代码和数据都在公共域中。

这是时区名称的来源,如America/Los_Angeles(和同义词,如US/Pacific)。

如果你需要跟踪不同的区域,那么你需要奥尔森数据库。正如其他人所建议的,您还希望以固定格式(UTC通常是所选的格式)存储数据,并记录生成数据的时区。您可能需要区分时间与UTC的偏移量和时区名称;这会在以后产生影响。此外,知道当前是2010-03-28T23:47:00-07:00(美国/太平洋)可能会或可能不会帮助您解释值2010-11-15T12:30-这可能是在PST(太平洋标准时间)而不是PDT(太平洋夏令时)中指定的。

标准的C库接口对这类东西没有太大的帮助。


奥尔森的数据发生了变化,部分原因是A·D·奥尔森很快就要退休了,另一部分原因是对维护者的版权侵权诉讼(现已驳回)。时区数据库现在由互联网号码分配机构IANA负责管理,并且在首页有一个“时区数据库”链接。讨论邮件列表现在是tz@iana.org; 公告列表是tz-announce@iana.org.

汤姆·斯科特(Tom Scott)在《计算机爱好者》频道的YouTube上关于时区的视频也对这一话题进行了精彩有趣的描述。示例包括:

萨摩亚(太平洋中的一个岛屿)将其时区提前24小时,以便于与澳大利亚和新西兰的贸易,西岸有两个人口在不同的时区,18世纪从儒略历到格里高利历的变化日历(发生在20世纪的俄罗斯)。

永远不要在没有UTC偏移(或时区参考)的情况下存储本地时间-如何不这样做的示例包括C¹中的FAT32和结构tm了解时区是一组UTC偏移(例如,冬季+0100,夏季+0200)切换发生的时间规则(可能会随着时间的推移而改变:例如,在20世纪90年代,欧盟将切换统一为3月和10月的最后一个星期日,标准时间02:00/夏令时03:00;之前成员国之间有所不同)。


¹struct tm的一些实现确实存储UTC偏移量,但这并未将其纳入标准。