3


我想只使用 Python 3.9 中的标准库来构建 DST 有效的时间戳,并希望在这个版本中可以做到这一点。

在我的时区“欧洲/柏林”中,2020 年的 DST 过境点是:2020 年
3 月 29 日 02:00 时钟切换到 03:00(没有 2 小时!)
2020 年 10 月 25 日 03:00时钟切换回 02:00(小时 2 存在两次!)

我的脚本产生以下输出:
MARCH
2020-03-29 01:59:00+01:00 CET plus 1 h: 2020-03-29 02: 59:00+01:00 CET
(应该是 03:59:00 CEST,因为没有 2 小时!)

OCTOBER
2020-10-25 02:00:00+02:00 CEST 加 1 小时:2020-10-25 03:00:00+01:00 CET
(似乎可以 - 编辑:应该是 02:00 CET!!!)

下面提供了示例代码。Windows 用户可能需要“pip install tzdata”才能使其工作。

任何建议将不胜感激!

'''
Should work out of the box with Python 3.9

Got a fallback import statement.

BACKPORT (3.6+)
pip install backports.zoneinfo

WINDOWS (TM) needs:
pip install tzdata
'''

from datetime import datetime, timedelta
from time import tzname
try:
    from zoneinfo import ZoneInfo
except ImportError:
    from backports import zoneinfo
    ZoneInfo = zoneinfo.ZoneInfo


tz = ZoneInfo("Europe/Berlin")
hour = timedelta(hours=1)

print("MARCH")
dt_01 = datetime(2020, 3, 29, 1, 59, tzinfo=tz)

dt_02 = dt_01 + hour
print(f"{dt_01} {dt_01.tzname()} plus 1 h: {dt_02} {dt_02.tzname()}")


print("\nOCTOBER")
dt_01 = datetime(2020, 10, 25, 2, 0, tzinfo=tz)
dt_02 = dt_01 + hour
print(f"{dt_01} {dt_01.tzname()} plus 1 h: {dt_02} {dt_02.tzname()}")
4

1 回答 1

4

尽管这是违反直觉的,但这是意料之中的。有关日期时间算法如何工作的更多详细信息,请参阅此博客文章。这样做的原因是,将 a 添加timedelta到 adatetime应该被认为是“将日历/时钟提前 X 量”,而不是“在这段时间过去后日历/时钟会说什么”。请注意,第一个问题可能会导致本地时区甚至不存在的时间!

如果你想,“什么datetime代表在这个代表的时间量过去之后它将是什么时间timedelta?” (您似乎这样做了),您应该做一些相当于转换为 UTC 并返回的事情,如下所示:

from datetime import datetime, timedelta, timezone

def absolute_add(dt: datetime, td: timedelta) -> datetime:
    utc_in = dt.astimezone(timezone.utc)  # Convert input to UTC
    utc_out = utc_in + td  # Do addition in UTC
    civil_out = utc_out.astimezone(dt.tzinfo)  # Back to original tzinfo
    return civil_out

我相信您可以创建一个timedelta覆盖的子类__add__来为您执行此操作(如果可以的话,我想在标准库中引入类似的东西)。

请注意,如果dt.tzinfoNone,这将使用您的系统本地时区来确定如何进行绝对加法,并将返回一个感知时区。运行这个America/New_York

>>> absolute_add(datetime(2020, 11, 1, 1), timedelta(hours=1))
datetime.datetime(2020, 11, 1, 1, 0, tzinfo=datetime.timezone(datetime.timedelta(days=-1, seconds=68400), 'EST'))

如果您希望这对天真的日期时间进行民事加法并为有意识的日期时间进行绝对加法,您可以检查它是否在函数中是天真的:

def absolute_add_nolocal(dt: datetime, td: timedelta) -> datetime:
    if dt.tzinfo is None:
        return dt + td
    return absolute_add(dt, td)

另外,需要明确的是,这与zoneinfo. 这一直是 Python 中日期时间的语义,我们无法以向后兼容的方式对其进行更改。pytz确实有点不同,因为添加pytz-aware datetimes 会做错事,并且normalize在算术发生后需要一个步骤,并且pytz作者决定normalize应该使用绝对时间语义。

absolute_add也适用于pytzand dateutil,因为它使用适用于所有时区库的操作。

于 2020-10-20T20:12:42.387 回答