Python datetime 用时区解析时间戳

时间:2021-05-02 21:11:20

标签: python-3.x datetime timestamp timezone

不久前我遇到了这个问题,其中解析带有时区数据的 ISO 字符串和解析时间戳(应该是同一时间)会给出不同的结果。我编写了一个测试来检查这种行为,它似乎非常不一致:

from pytz import timezone as tz
from datetime import datetime

timezone = "Australia/Sydney"
start_time = "2021-05-04T08:12:00"
tz_object = tz(timezone)
naive_datetime = datetime.fromisoformat(start_time)


zoned_time = datetime(naive_datetime.year, naive_datetime.month, naive_datetime.day, naive_datetime.hour, naive_datetime.minute, naive_datetime.second, tzinfo=tz_object)
parsed_time = datetime.fromtimestamp(zoned_time.timestamp(), tz_object)

assert zoned_time.time() == naive_datetime.time()
assert zoned_time.time() == parsed_time.time()

此测试产生以下输出

  File "test.py", line 13, in <module>
    assert zoned_time.time() == parsed_time.time()
  AssertionError

除非我遗漏了一些东西,否则我只能得出结论,这条线产生的时间

parsed_time = datetime.fromtimestamp(zoned_time.timestamp(), tz_object)

产生与解析实际 ISO 字符串不同的时间。通常,我希望解析时间的时间戳返回一个时间戳,该时间戳指的是分配给它的时区中的上午 8 点 12 分。

这种行为是预期的吗? 我错过了什么吗?

1 个答案:

答案 0 :(得分:1)

在 Python 3.9 中,使用 zoneinfo。请注意,如果需要,pytz 有一个弃用垫片。

然后您的代码可以正常工作

from datetime import datetime
from zoneinfo import ZoneInfo

timezone = "Australia/Sydney"
tz_object = ZoneInfo(timezone)

start_time = "2021-05-04T08:12:00"
naive_datetime = datetime.fromisoformat(start_time)

zoned_time = datetime(naive_datetime.year, naive_datetime.month, naive_datetime.day, naive_datetime.hour, naive_datetime.minute, naive_datetime.second, tzinfo=tz_object)
parsed_time = datetime.fromtimestamp(zoned_time.timestamp(), tz_object)

assert zoned_time.time() == naive_datetime.time()
assert zoned_time.time() == parsed_time.time()

至于为什么出现断言错误:使用 pytz 您需要 localize 带有时区的日期时间对象; 切勿直接使用 pytz 设置 tzinfo