Python:给定时区名称的所有可能的时区缩写(反之亦然)

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)

我正在寻找的是任何时区缩写,例如PSTPDT,忽略当前日期时间(例如现在),返回所有可能的时区名称,在这种情况下是一个包​​含America/Los_Angeles的列表。

谢谢

unu*_*tbu 6

由于您希望忽略当前日期时间,因此听起来您想查找过去任何时间曾使用给定缩写的任何时区。该信息位于 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/ShanghaiUS/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)

  • 值得注意的是,这些并不一定实际上都指同一个区域。考虑“tzones['CST']”包括“古巴”、“亚洲/上海”和“美国/中部” (2认同)

Han*_*600 5

我喜欢 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)


GG_*_*hon 5

更新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)。

  • 注意:对于给定的地理时区,由于 DST 有效或无效,一年中可能会应用不同的 UTC 偏移量。因此,使用“datetime.now()”(无需使用 [utcnow](https://blog.ganssle.io/articles/2019/11/utcnow.html) 顺便说一句。)只提供了完整图片的一部分(因此只是所有缩写的一小部分)。 (2认同)