我有一个Django应用程序,我有一个硬编码的时区列表。我知道pytz
可以用来获取所有时区的列表。但它并没有显示它们相对于格林威治标准时间的时差。请建议如何获得这样的列表。
TIMEZONE_CHOICES = (
("<DstTzInfo 'Africa/Abidjan' LMT-1 day, 23:44:00 STD>", "<DstTzInfo 'Africa/Abidjan' LMT-1 day, 23:44:00 STD>"),
("<DstTzInfo 'Africa/Accra' LMT-1 day, 23:59:00 STD>", "<DstTzInfo 'Africa/Accra' LMT-1 day, 23:59:00 STD>"),
("<DstTzInfo 'Africa/Addis_Ababa' LMT+2:27:00 STD>", "<DstTzInfo 'Africa/Addis_Ababa' LMT+2:27:00 STD>"),
("<DstTzInfo 'Africa/Algiers' LMT+0:12:00 STD>", "<DstTzInfo 'Africa/Algiers' LMT+0:12:00 STD>"),
("<DstTzInfo 'Africa/Asmara' LMT+2:27:00 STD>", "<DstTzInfo 'Africa/Asmara' LMT+2:27:00 STD>"),...)
假设与 UTC 的时间差是 GMT 的可接受替代方法(UTC 和 GMT 几乎但不完全相同),则可以从pytz.timezone
对象获取时区偏移量和相关信息。
对于具有多个偏移量的时区(由于夏令时、历史更改等),您可以从_tzinfos
属性中获取相关的区域和偏移量。对于具有单个偏移量的时区,您可以从_utcoffset
属性中获取偏移量。
以下示例创建一个字典,其中pytz.all_timezones
中的每个时区都是一个键,值是包含相关区域和偏移量的元组列表。
from pytz import all_timezones, timezone
timezones = {}
for tz in all_timezones:
tzinfos = getattr(timezone(tz), '_tzinfos', None)
if tzinfos:
timezones[tz] = [(zone, str(offset)) for offset, dst, zone in tzinfos]
else:
timezones[tz] = [(tz, str(timezone(tz)._utcoffset))]
print(timezones)
# {
# 'Africa/Abidjan': [('LMT', '-1 day, 23:44:00'), ('GMT', '0:00:00')],
# 'Africa/Accra': [('LMT', '-1 day, 23:59:00'), ('GMT', '0:00:00'), ('+0020', '0:20:00')],
# 'Africa/Addis_Ababa': [('LMT', '2:27:00'), ('EAT', '3:00:00'), ('+0230', '2:30:00'), ('+0245', '2:45:00')],
# 'Africa/Algiers': [('LMT', '0:12:00'), ('PMT', '0:09:00'), ('WET', '0:00:00'), ('WEST', '1:00:00'), ('CET', '1:00:00'), ('CEST', '2:00:00')],
# 'Africa/Asmara': [('LMT', '2:27:00'), ('EAT', '3:00:00'), ('+0230', '2:30:00'), ('+0245', '2:45:00')],
# ...
# }
如果您更喜欢将每个区域的信息联接到字符串中的列表(如您的示例),则可以按如下方式进行修改。
timezones = []
for tz in all_timezones:
tzinfos = getattr(timezone(tz), '_tzinfos', None)
if tzinfos:
timezones.extend([' '.join([tz, zone, str(offset)]) for offset, dst, zone in tzinfos])
else:
timezones.append(' '.join([tz, str(timezone(tz)._utcoffset)]))
根据您关于删除偏移量不在输出一刻钟间隔的某个位置的时区的评论,您可能希望删除历史日期(通常是 19 世纪和 20 世纪初日期)中包含的各种时区,其中大多数都包含在某种"平均时间"变体中,例如"LMT"表示"本地平均时间"。删除这些内容的一种蛮力方法是过滤掉除"GMT"以外的所有以"MT"结尾的时区缩写。目前使用中可能还有其他一些我不知道的例外情况,但也应该以与"GMT"相同的方式处理。
timezones = {}
for tz in all_timezones:
tzinfos = getattr(timezone(tz), '_tzinfos', None)
if tzinfos:
timezones[tz] = [(zone, str(offset)) for offset, dst, zone in tzinfos if zone == 'GMT' or not zone.endswith('MT')]
else:
timezones[tz] = [(tz, str(timezone(tz)._utcoffset))]
print(timezones)
# {
# 'Africa/Abidjan': [('GMT', '0:00:00')],
# 'Africa/Accra': [('GMT', '0:00:00'), ('+0020', '0:20:00')],
# 'Africa/Addis_Ababa': [('EAT', '3:00:00'), ('+0230', '2:30:00'), ('+0245', '2:45:00')],
# 'Africa/Algiers': [('WET', '0:00:00'), ('WEST', '1:00:00'), ('CET', '1:00:00'), ('CEST', '2:00:00')],
# 'Africa/Asmara': [('EAT', '3:00:00'), ('+0230', '2:30:00'), ('+0245', '2:45:00')],
# ...
# }