jef*_*tle 5 python timezone datetime pytz
使用pytz
,我知道如何列出时区名称,但我想获得每个时区名称的所有可能的时区缩写:
import pytz
list(pytz.common_timezones)
['Africa/Abidjan', 'Africa/Accra', 'Africa/Addis_Ababa',...]
Run Code Online (Sandbox Code Playgroud)
我正在寻找的是任何时区缩写,例如PST或PDT,忽略当前日期时间(例如现在),返回所有可能的时区名称,在这种情况下是一个包含America/Los_Angeles的列表。
谢谢
由于您希望忽略当前日期时间,因此听起来您想查找过去任何时间曾使用给定缩写的任何时区。该信息位于 Olson 数据库中,可通过 pytz 访问。但是,pytz 将此信息存储在私有属性中
tzone._transition_info
:
import collections
import datetime as DT
import pytz
tzones = collections.defaultdict(set)
abbrevs = collections.defaultdict(set)
for name in pytz.all_timezones:
tzone = pytz.timezone(name)
for utcoffset, dstoffset, tzabbrev in getattr(
tzone, '_transition_info', [[None, None, DT.datetime.now(tzone).tzname()]]):
tzones[tzabbrev].add(name)
abbrevs[name].add(tzabbrev)
Run Code Online (Sandbox Code Playgroud)
第三个(默认)参数 to 的原因gettattr
是处理几个时区,例如Africa/Bujumbura
从未有任何转换的时区。所以这些情况下的缩写就是当前的缩写。
In [94]: tzones['PST']
Out[94]:
{'America/Bahia_Banderas',
'America/Boise',
'America/Creston',
'America/Dawson',
'America/Dawson_Creek',
'America/Ensenada',
'America/Hermosillo',
'America/Inuvik',
'America/Juneau',
'America/Los_Angeles',
'America/Mazatlan',
'America/Metlakatla',
'America/Santa_Isabel',
'America/Sitka',
'America/Tijuana',
'America/Vancouver',
'America/Whitehorse',
'Canada/Pacific',
'Canada/Yukon',
'Mexico/BajaNorte',
'Mexico/BajaSur',
'PST8PDT',
'Pacific/Pitcairn',
'US/Pacific',
'US/Pacific-New'}
Run Code Online (Sandbox Code Playgroud)
In [95]: tzones['PDT']
Out[95]:
{'America/Boise',
'America/Dawson',
'America/Dawson_Creek',
'America/Ensenada',
'America/Juneau',
'America/Los_Angeles',
'America/Metlakatla',
'America/Santa_Isabel',
'America/Sitka',
'America/Tijuana',
'America/Vancouver',
'America/Whitehorse',
'Canada/Pacific',
'Canada/Yukon',
'Mexico/BajaNorte',
'PST8PDT',
'US/Pacific',
'US/Pacific-New'}
Run Code Online (Sandbox Code Playgroud)
In [97]: abbrevs['America/Los_Angeles']
Out[97]: {'LMT', 'PDT', 'PPT', 'PST', 'PWT'}
Run Code Online (Sandbox Code Playgroud)
正如Paul 指出的那样,请注意时区缩写是不明确的——它们不一定映射到具有相同 utcoffset 的时区。例如,无论是Asia/Shanghai
与US/Central
使用的CST
时区缩写。
In [242]: 'Asia/Shanghai' in tzones['CST']
Out[242]: True
In [243]: 'US/Central' in tzones['CST']
Out[243]: True
Run Code Online (Sandbox Code Playgroud)
我喜欢 unutbu 的答案,它让我找到了对我有用的答案。我有用户的区域设置,所以我发现我可以用它来消除时区缩写之间的歧义。
换句话说,这解决了以下问题:
In [242]: 'Asia/Shanghai' in tzones['CST']
Out[242]: True
In [243]: 'US/Central' in tzones['CST']
Out[243]: True
Run Code Online (Sandbox Code Playgroud)
此函数需要两个字母的国家代码:
def GetTimeZoneName(timezone, country_code):
#see if it's already a valid time zone name
if timezone in pytz.all_timezones:
return timezone
#if it's a number value, then use the Etc/GMT code
try:
offset = int(timezone)
if offset > 0:
offset = '+' + str(offset)
else:
offset = str(offset)
return 'Etc/GMT' + offset
except ValueError:
pass
#look up the abbreviation
country_tzones = None
try:
country_tzones = pytz.country_timezones[country_code]
except:
pass
set_zones = set()
if country_tzones is not None and len(country_tzones) > 0:
for name in country_tzones:
tzone = pytz.timezone(name)
for utcoffset, dstoffset, tzabbrev in getattr(tzone, '_transition_info', [[None, None, datetime.datetime.now(tzone).tzname()]]):
if tzabbrev.upper() == timezone.upper():
set_zones.add(name)
if len(set_zones) > 0:
return min(set_zones, key=len)
# none matched, at least pick one in the right country
return min(country_tzones, key=len)
#invalid country, just try to match the timezone abbreviation to any time zone
for name in pytz.all_timezones:
tzone = pytz.timezone(name)
for utcoffset, dstoffset, tzabbrev in getattr(tzone, '_transition_info', [[None, None, datetime.datetime.now(tzone).tzname()]]):
if tzabbrev.upper() == timezone.upper():
set_zones.add(name)
return min(set_zones, key=len)
Run Code Online (Sandbox Code Playgroud)
这将返回 CST 的正确时区:
>>> GetTimeZoneName('CST','CN')
'Asia/Shanghai'
>>> GetTimeZoneName('CST','US')
'America/Detroit'
Run Code Online (Sandbox Code Playgroud)
更新py3.9+ (如PEP615中所述),新的zoneinfo模块可能会有所帮助:
from collections import defaultdict
from datetime import datetime as dt
from zoneinfo import available_timezones, ZoneInfo
now = dt.utcnow()
tz_key = lambda tz: ZoneInfo(tz).tzname(now)
tz_map = defaultdict(list)
for tz in available_timezones():
tz_map[tz_key(tz)].append(tz)
tz_map = {k: sorted(v) for k, v in tz_map.items()}
Run Code Online (Sandbox Code Playgroud)
例如,print(tz_map['PDT'])
会产生:['America/Ensenada', 'America/Los_Angeles', 'America/Santa_Isabel', 'America/Tijuana', 'America/Vancouver', 'Canada/Pacific', 'Mexico/BajaNorte', 'PST8PDT', 'US/Pacific', 'US/Pacific-New']
注意:时区取自本地安装的时区数据。您还可以添加第一方 tzdata 库(不在标准库中,但由 python 的核心开发人员维护: use pip install tzdata
)。
归档时间: |
|
查看次数: |
9889 次 |
最近记录: |