Python使用技巧--时区的转换

技巧:用datetime模块处理时区转换,不要用time模块

time模块

time模块只能转换本地时区的时间,不能转换其他时区的时间

比如以下例子对本地时区(CST)进行转换,程序能运行:

time_format = "%Y-%m-%d %H:%M:%S %Z"
time_str = "2021-09-25 15:49:15 CST"
# 将字符串格式转换成时间结构格式
time_tuple = time.strptime(time_str, time_format)
time_str = time.strftime(time_format, time_tuple)
print(time_str)
# "2021-09-25 15:49:15 CST"

但是如果将CST改成其他时区,比如“EDT”,则程序报错。

time_format = "%Y-%m-%d %H:%M:%S %Z"
time_str = "2021-09-25 15:49:15 EDT"
# 将字符串格式转换成时间结构格式
time_tuple = time.strptime(time_str, time_format)
time_str = time.strftime(time_format, time_tuple)
print(time_str)
# ValueError: time data '2021-09-25 15:49:15 EDT' does not match format '%Y-%m-%d %H:%M:%S %Z'

之所以出现这个问题,是因为time模块本质上仍然要依赖具体的平台而运作。它的行为取决于底层的C函数与宿主操作系统之间的协作方式,这导致该模块没办法稳定地处理多个时区,所以不要用这个模块来编写时区转换这方面的代码。如果一定要使用,那最多使用在UTC时间和宿主计算机的当地时区之间的转换,涉及其他时区的转换操作,最好还是通过datetime模块来做。

datetime模块

使用datetime模块时将GMT(UTC+0)时间转换成本地时间(CST时区,比GMT时区多8个小时):

time_format = "%Y-%m-%d %H:%M:%S"
time_str = "2021-09-25 15:49:15"
# 将字符串时间格式转换成datetime格式
now = datetime.datetime.strptime(time_str, time_format)
now_utc = now.replace(tzinfo=datetime.timezone.utc)
now_local = now_utc.astimezone()
print(now_local)
# 2021-09-25 23:49:15+08:00

当使用datetime模块时,可以把一个时区的本地时间可靠地转化成另一个时区的本地时间。但需要配合pytz模块进行使用。因为datetime的这套时区操作机制必须通过tzinfo类与相关的方法来运作,而系统在安装python的时候,并不会默认安装UTC之外的时区定义信息。

pytz的安装,可通过pip进行安装。一般把一个时区的本地时间转化成另一个时区的本地时间的步骤是:

  1. 把要转换的时间转换成GMT(UTC+0)时间
  2. 把GMT时间转化成另一个时区的本地时间

把要转换的时区("Asia/Shanghai")转换成GMT时区

import pytz
import datetime

time_str = "2021-09-25 15:49:15"
# 将字符串时间格式转换成datetime形式
old_dt = datetime.datetime.strptime(time_str, time_format)

# 将源时区的datetime形式转换成GMT时区的datetime形式
dt = pytz.timezone("Asia/Shanghai").localize(old_dt)
utc_dt = pytz.utc.normalize(dt.astimezone(pytz.utc))
print(utc_dt)
# 2021-09-25 15:49:15+00:00

把GMT时间转化成另一个时区("Europe/London")的本地时间

_timezone = pytz.timezone("Europe/London")
new_dt = _timezone.normalize(utc_dt.astimezone(_timezone))
print(new_dt)
# 2021-09-25 08:49:15-07:00

以下是封装好的方法:

import datetime
import pytz

def timezone_change(time_str, src_timezone, dst_timezone=None, time_format=None):
    """
    将任一时区的时间转换成指定时区的时间
    如果没有指定目的时区,则默认转换成当地时区

    :param time_str:
    :param src_timezone: 要转换的源时区,如"Asia/Shanghai", "UTC"
    :param dst_timezone: 要转换的目的时区,如"Asia/Shanghai", 如果没有指定目的时区,则默认转换成当地时区
    :param dst_timezone: 时间格式
    :return: str, 字符串时间格式
    """
    if not time_format:
        time_format = "%Y-%m-%d %H:%M:%S"

    # 将字符串时间格式转换成datetime形式
    old_dt = datetime.datetime.strptime(time_str, time_format)

    # 将源时区的datetime形式转换成GMT时区(UTC+0)的datetime形式
    dt = pytz.timezone(src_timezone).localize(old_dt)
    utc_dt = pytz.utc.normalize(dt.astimezone(pytz.utc))

    # 将GMT时区的datetime形式转换成指定的目的时区的datetime形式
    if dst_timezone:
        _timezone = pytz.timezone(dst_timezone)
        new_dt = _timezone.normalize(utc_dt.astimezone(_timezone))
    else:
        # 未指定目的时间,默认转换成当地时区
        new_dt = utc_dt.astimezone()
    # 转换成字符串时间格式
    return new_dt.strftime(time_format)

方法的使用:

print(timezone_change("2021-09-25 15:49:15", src_timezone="Asia/Shanghai", dst_timezone="Europe/London"))

# 北京时间比伦敦时间多7小时
# 2021-09-25 08:49:15

posted on 2021-09-25 18:20  xufat  阅读(5445)  评论(0编辑  收藏  举报

导航

/* 返回顶部代码 */ TOP