gpt4 book ai didi

python - 使用 pytz 从 UTC 偏移 "Canonical"?

转载 作者:太空宇宙 更新时间:2023-11-03 11:28:57 26 4
gpt4 key购买 nike

我很好奇如何调用某些时区选择器上提供的“规范”时区偏移量(如果有这样的东西,作为规范偏移量,我什至不确定)。

例如,在 Windows XP 上,您可以看到对于 东部时间(美国和加拿大),下拉列表始终显示 GMT-05:00 这实际上是全年不正确,因为当夏令时生效时,与 UTC 的偏移量仅为 -4:00。另一方面,每个人都提到 US/Eastern 与 UTC 相差 5 小时。我想知道 -5:00 是如何调用的。电子邮件时间标识符?规范时区偏移量?

另外,如果我用 pytz 创建一个 US/Eastern 时区,有没有一种方法可以获取 -5:00 而不管实际的 now 时间是否在夏令时?我的意思是...我想知道是否有一个函数,或者...可以得到 -5:00 的东西,无论我是在今天还是在 8 月中旬运行它(当 DST 启用且实际偏移量仅为 -4:00 时)

Windows XP timezone selector

图片来自http://www.microsoft.com/library/media/1033/windowsxp/images/using/setup/tips/67446-change-time-zone.gif

提前谢谢你。

最佳答案

如果我们将“规范”表示不在 DST 中的日期的 utcoffset,那么问题就简化为查找不在 DST 中的日期(针对每个时区)。

我们可以先试试当前日期。如果不是 DST,那么我们很幸运。如果是,那么我们可以遍历 utc 转换日期列表(存储在 tzone._utc_transition_times 中),直到找到一个不是 DST 的日期:

import pytz
import datetime as DT
utcnow = DT.datetime.utcnow()

canonical = dict()
for name in pytz.all_timezones:
tzone = pytz.timezone(name)
try:
dstoffset = tzone.dst(utcnow, is_dst=False)
except TypeError:
# pytz.utc.dst does not have a is_dst keyword argument
dstoffset = tzone.dst(utcnow)
if dstoffset == DT.timedelta(0):
# utcnow happens to be in a non-DST period
canonical[name] = tzone.localize(utcnow, is_dst=False).strftime('%z')
else:
# step through the transition times until we find a non-DST datetime
for transition in tzone._utc_transition_times[::-1]:
dstoffset = tzone.dst(transition, is_dst=False)
if dstoffset == DT.timedelta(0):
canonical[name] = (tzone.localize(transition, is_dst=False)
.strftime('%z'))
break

for name, utcoffset in canonical.iteritems():
print('{} --> {}'.format(name, utcoffset))

# All timezones have been accounted for
assert len(canonical) == len(pytz.all_timezones)

产量

...
Mexico/BajaNorte --> -0800
Africa/Kigali --> +0200
Brazil/West --> -0400
America/Grand_Turk --> -0400
Mexico/BajaSur --> -0700
Canada/Central --> -0600
Africa/Lagos --> +0100
GMT-0 --> +0000
Europe/Sofia --> +0200
Singapore --> +0800
Africa/Tripoli --> +0200
America/Anchorage --> -0900
Pacific/Nauru --> +1200

请注意,上面的代码访问私有(private)属性 tzone._utc_transition_times。这是 pytz 中的一个实现细节。由于它不是公共(public) API 的一部分,因此不能保证它存在于 pytz 的 future 版本中。事实上,它甚至不存在于当前版本的 pytz 中的所有时区——特别是,它不存在于没有 DST 转换时间的时区,例如 'Africa/Bujumbura' . (这就是为什么我会先检查 utcnow 是否恰好处于非 DST 时间段。)

如果您想要一种不依赖私有(private)属性的方法,我们可以简单地将 utcnow 前退一天,直到我们找到非 DST 时间段内的一天。该代码会比上面的代码慢一点,但由于您实际上只需要运行此代码一次即可收集所需的信息,所以这应该无关紧要。

下面是不使用 _utc_transition_times 时的代码:

import pytz
import datetime as DT
utcnow = DT.datetime.utcnow()

canonical = dict()
for name in pytz.all_timezones:
tzone = pytz.timezone(name)
try:
dstoffset = tzone.dst(utcnow, is_dst=False)
except TypeError:
# pytz.utc.dst does not have a is_dst keyword argument
dstoffset = tzone.dst(utcnow)
if dstoffset == DT.timedelta(0):
# utcnow happens to be in a non-DST period
canonical[name] = tzone.localize(utcnow, is_dst=False).strftime('%z')
else:
# step through the transition times until we find a non-DST datetime
date = utcnow
while True:
date = date - DT.timedelta(days=1)
dstoffset = tzone.dst(date, is_dst=False)
if dstoffset == DT.timedelta(0):
canonical[name] = (tzone.localize(date, is_dst=False)
.strftime('%z'))
break

for name, utcoffset in canonical.iteritems():
print('{} --> {}'.format(name, utcoffset))

# All timezones have been accounted for
assert len(canonical) == len(pytz.all_timezones)

关于python - 使用 pytz 从 UTC 偏移 "Canonical"?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27491988/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com