6

在 Python 中,我可以找到本地时间的 Unix 时间戳,知道时区,就像这样(使用 pytz):

>>> import datetime as DT
>>> import pytz
>>> mtl = pytz.timezone('America/Montreal')
>>> naive_time3 = DT.datetime.strptime('2013/11/03', '%Y/%m/%d')
>>> naive_time3
datetime.datetime(2013, 11, 3, 0, 0)
>>> localized_time3 = mtl.localize(naive_time3)
>>> localized_time3
datetime.datetime(2013, 11, 3, 0, 0, tzinfo=<DstTzInfo 'America/Montreal' EDT-1 day, 20:00:00 DST>)
>>> localized_time3.timestamp()
1383451200.0

到目前为止,一切都很好。naive_time不知道时区,而知道其在 Montréallocalized_time的 2013/11/03 的午夜,因此 (UTC) Unix 时间戳很好。这个时区也是我的本地时区,这个时间戳似乎是正确的:

$ date -d @1383451200
Sun Nov  3 00:00:00 EDT 2013

现在,蒙特利尔的时钟在 11 月 3 日下午 2:00 后调了一小时,所以那天我们多出了一个小时。这意味着在 2013/11/03 和 2013/11/04 之间有25 小时。这表明:

>>> naive_time4 = DT.datetime.strptime('2013/11/04', '%Y/%m/%d')
>>> localized_time4 = mtl.localize(naive_time4)
>>> localized_time4
datetime.datetime(2013, 11, 4, 0, 0, tzinfo=<DstTzInfo 'America/Montreal' EST-1 day, 19:00:00 STD>)
>>> (localized_time4.timestamp() - localized_time3.timestamp()) / 3600
25.0

现在,我正在寻找一种localized_time4从 中获取对象的简单方法localized_time3,因为我知道我想在同一时间(此处为午夜)获取下一个本地化日。我试过timedelta了,但我相信它不知道时区或 DST:

>>> localized_time4td = localized_time3 + DT.timedelta(1)
>>> localized_time4td
datetime.datetime(2013, 11, 4, 0, 0, tzinfo=<DstTzInfo 'America/Montreal' EDT-1 day, 20:00:00 DST>)
>>> (localized_time4td.timestamp() - localized_time3.timestamp()) / 3600
24.0

我的目的是获取有关每个本地日的 Unix 时间戳存储的日志条目的信息。当然,如果我在这里使用localized_time3.timestamp()和添加24 * 3600(这将与 相同localized_time4td.timestamp()),我会错过所有发生在localized_time4td.timestamp()和之间的日志条目localized_time4td.timestamp() + 3600

换句话说,我正在寻找的函数或方法应该知道何时将 25 小时、24 小时或 23 小时添加到 Unix 时间戳,具体取决于 DST 转换发生的时间。

4

4 回答 4

3

不使用新包:

def add_day(x):
    d = x.date()+DT.timedelta(1)
    return mtl.localize(x.replace(year=d.year, month=d.month, day=d.day, tzinfo=None))

完整脚本:

import datetime as DT
import pytz
import calendar
mtl = pytz.timezone('America/Montreal')
naive_time3 = DT.datetime.strptime('2013/11/03', '%Y/%m/%d')
print repr(naive_time3)
#datetime.datetime(2013, 11, 3, 0, 0)
localized_time3 = mtl.localize(naive_time3)
print repr(localized_time3)
#datetime.datetime(2013, 11, 3, 0, 0, tzinfo=<DstTzInfo 'America/Montreal' EDT-1 day, 20:00:00 DST>)
print calendar.timegm(localized_time3.utctimetuple())
#1383451200.0
def add_day(x):
    d = x.date()+DT.timedelta(1)
    return mtl.localize(x.replace(year=d.year, month=d.month, day=d.day, tzinfo=None))
print repr(add_day(localized_time3))
#datetime.datetime(2013, 11, 4, 0, 0, tzinfo=<DstTzInfo 'America/Montreal' EST-1 day, 19:00:00 STD>)

calendar适用于 Python2。)

于 2013-11-29T02:16:30.933 回答
2

在此答案的最后,我逐渐提供了几种解决方案,其中最强大的解决方案试图处理以下问题:

  • 由于 DST 导致的 UTC 偏移
  • 由于与 DST 无关的原因,本地时区可能具有不同 UTC 偏移量的过去日期。dateutil和 stdlib 解决方案在某些系统上失败,特别是 Windows
  • DST期间的模棱两可(不知道是否Arrow提供接口来处理它)
  • DST 期间不存在的时间(相同)

要在给定时区查找明天午夜(或其他固定时间)的 POSIX 时间戳,您可以使用如何获取给定时区的“午夜”的 UTC 时间?

from datetime import datetime, time, timedelta
import pytz

DAY = timedelta(1)
tz = pytz.timezone('America/Montreal')

tomorrow = datetime(2013, 11, 3).date() + DAY

midnight = tz.localize(datetime.combine(tomorrow, time(0, 0)), is_dst=None)
timestamp = (midnight - datetime(1970, 1, 1, tzinfo=pytz.utc)).total_seconds()

dt.date()dt方法为幼稚和时区感知对象返回相同的幼稚日期。

时间戳的显式公式用于支持 Python 3.3 之前的 Python 版本。否则.timestamp()方法可以在 Python 3.3+ 中使用。

为了避免在 DST 转换期间解析输入日期时出现歧义,.localize()除非您知道is_dst参数,否则方法不可避免,您可以使用与日期一起存储的 Unix 时间戳:

from datetime import datetime, time, timedelta
import pytz

DAY = timedelta(1)
tz = pytz.timezone('America/Montreal')

local_dt = datetime.fromtimestamp(timestamp_from_the_log, tz)
tomorrow = local_dt.date() + DAY

midnight = tz.localize(datetime.combine(tomorrow, time(0, 0)), is_dst=None)
timestamp = (midnight - datetime(1970, 1, 1, tzinfo=pytz.utc)).total_seconds()

支持其他固定时间(不仅是午夜):

tomorrow = local_dt.replace(tzinfo=None) + DAY # tomorrow, same time
dt_plus_day = tz.localize(tomorrow, is_dst=None)
timestamp = dt_plus_day.timestamp() # use the explicit formula before Python 3.3

is_dst=None如果结果日期不明确或不存在,则引发异常。为避免异常,您可以选择最接近昨天的前一个日期的时间(相同的 DST 状态,即is_dst=local_dt.dst()):

from datetime import datetime, time, timedelta
import pytz

DAY = timedelta(1)
tz = pytz.timezone('America/Montreal')

local_dt = datetime.fromtimestamp(timestamp_from_the_log, tz)
tomorrow = local_dt.replace(tzinfo=None) + DAY

dt_plus_day = tz.localize(tomorrow, is_dst=local_dt.dst())
dt_plus_day = tz.normalize(dt_plus_day) # to detect non-existent times                                            
timestamp = (dt_plus_day - datetime(1970, 1, 1, tzinfo=pytz.utc)).total_seconds()

.localize()尊重给定的时间,即使它不存在,因此.normalize()需要确定时间。normalize()如果方法更改其输入(在这种情况下检测到不存在的时间)以与其他代码示例保持一致,您可以在此处引发异常。

于 2013-12-02T17:01:51.130 回答
1

(感谢@rdodev 将我指向Arrow)。

使用 Arrow,此操作变得简单:

>>> import arrow
>>> import datetime as DT
>>> lt3 = arrow.get(DT.datetime(2013, 11, 3), 'America/Montreal')
>>> lt3
<Arrow [2013-11-03T00:00:00-04:00]>
>>> lt4 = arrow.get(DT.datetime(2013, 11, 4), 'America/Montreal')
>>> lt4
<Arrow [2013-11-04T00:00:00-05:00]>
>>> lt4.timestamp - (lt3.replace(days=1).timestamp)
0
>>> (lt3.replace(days=1).timestamp - lt3.timestamp) / 3600
25.0

使用 Arrow 的replace方法,单数单位名称替换该属性,而复数添加它。lt3.replace(days=1)2013 年 11 月 4 日和 2013 年11 月 1 日也是如此lt3.replace(day=1)

于 2013-11-29T01:21:21.750 回答
1

这里的替代方案基于dateutil

>>> # In Spain we changed DST 10/26/2013
>>> import datetime
>>> import dateutil.tz
>>> # tzlocal gets the timezone of the computer
>>> dt1 = datetime.datetime(2013, 10, 26, 14, 00).replace(tzinfo=dateutil.tz.tzlocal())
>>> print dt1
2013-10-26 14:00:00+02:00
>>> dt2 = dt1 + datetime.timedelta(1)
>>> print dt2
2013-10-27 14:00:00+01:00
# see if we hace 25 hours of difference
>>> import time
>>> (time.mktime(dt2.timetuple()) - time.mktime(dt1.timetuple())) / 3600.0
25.0
>>> (float(dt2.strftime('%s')) - float(dt1.strftime('%s'))) / 3600   # the same
25.0
于 2013-11-29T03:10:15.903 回答