3

我想计算日期间隔中有多少小时:例如“2014.03.29-30”应该给出 47,因为夏令时。

我的方法是制作两个日期时间对象,在以下示例中:

datetime.datetime(2014, 3, 29, 0, 0, tzinfo=<DstTzInfo 'Europe/Budapest' LMT+1:16:00 STD>)
datetime.datetime(2014, 3, 30, 23, 59, tzinfo=<DstTzInfo 'Europe/Budapest' LMT+1:16:00 STD>)
return (date2-date1) + timedelta(minutes=1)

但是,它给出了“2 天,0:00:00”,这是不正确的。如何制作一个考虑时区和 dst 的 timedelta 对象?此外,如果整个问题有一个更简单的解决方案,我愿意接受。

谢谢!

4

1 回答 1

3

在 1901-12-13 20:45:52 UTC 之前,'Europe/Budapest'时区为 LMT+1:16:00 STD。目前,截至 2016-05-05,'Europe/Budapest'时区为 CET+2:00:00 DST。

如果您使用 pytz 的localize方法,那么 pytz 将选择'Europe/Budapest'适合给定的原始日期时间的时区(utcoffset 和 dstoffset):

import datetime as DT
import pytz

tzone = pytz.timezone('Europe/Budapest')
date1 = tzone.localize(DT.datetime(2014, 3, 29, 0, 0), is_dst=None)
# datetime.datetime(2014, 3, 29, 0, 0, tzinfo=<DstTzInfo 'Europe/Budapest' CET+1:00:00 STD>)

相反,如果您tzinfo=tzone直接向 提供datetime.datetime,如下所示:

wrong_date1 = datetime.datetime(2014, 3, 29, 0, 0, tzinfo=tzone)
# datetime.datetime(2014, 3, 29, 0, 0, tzinfo=<DstTzInfo 'Europe/Budapest' LMT+1:16:00 STD>)

然后datetime.datetime 错误地选择与关联的第一个时区,'Europe/Budapest' 无论该时区是否是 2014-3-29 生效的时区

因此,在使用 pytz 时,请始终使用tzone.localizenaive datetimes timezone-aware:

import datetime as DT
import pytz
tzone = pytz.timezone('Europe/Budapest')
date1 = tzone.localize(DT.datetime(2014, 3, 29, 0, 0), is_dst=None)
date2 = tzone.localize(DT.datetime(2014, 3, 30, 23, 59), is_dst=None)
print(((date2-date1) + DT.timedelta(minutes=1)).total_seconds()/3600.)
# 47.0

不要使用tzinfo=tzone除非tzonepytz.utc(或在其整个历史中始终相同的时区。)


日期1901-12-13 20:45:52 UTC从何而来?

tzone._utc_transition_times您可以使用其和tzone._transition_info私有属性查看 pytz 时区的 utc 转换时间(以及相关的转换信息) :

In [43]: [(utcdate, utcoffset, dstoffset, tzabbrev) for utcdate, (utcoffset, dstoffset, tzabbrev) in zip(tzone._utc_transition_times, tzone._transition_info)][:2]
Out[43]: 
[(datetime.datetime(1, 1, 1, 0, 0),
  datetime.timedelta(0, 4560),
  datetime.timedelta(0),
  'LMT'),
 (datetime.datetime(1901, 12, 13, 20, 45, 52),
  datetime.timedelta(0, 3600),
  datetime.timedelta(0),
  'CET')]

这表明从日期1-1-1 UTC1901-12-13 20:45:52 UTC时区缩写是LMT,utcoffset 是 4560 秒,等于 1 小时 16 分钟:

In [47]: print(DT.timedelta(0, 4560))
1:16:00

因此,与关联的第一个时区'Europe/Budapest'LMT+1:16:00 STD

于 2016-05-05T16:36:30.580 回答