11

delorean文档显示了这种获取给定时区当前时间的 方法datetime

from datetime import datetime
from pytz import timezone

EST = "US/Eastern"
UTC = "UTC"

d = datetime.utcnow()
utc = timezone(UTC)
est = timezone(EST)
d = utc.localize(d)
d = est.normalize(EST)

并将其与基于 delorian 的代码进行比较:

from delorean import Delorean

EST = "US/Eastern"

d = Delorean(timezone=EST)

我相信这个datetime例子应该写成:

from datetime import datetime
import pytz

eastern_timezone = pytz.timezone("US/Eastern")
d = datetime.now(eastern_timezone)

那更简洁。

是否存在最后一个代码示例失败而第一个代码示例继续工作的情况?


更新: 当前示例:

from datetime import datetime
import pytz

d = datetime.utcnow()
d = pytz.utc.localize(d)

est = pytz.timezone('US/Eastern')
d = est.normalize(d)
return d

那还是太冗长了。

问题仍然存在:您是否需要通过 utc 进行显式往返,tz.normalize()或者您可以使用它datetime.now(tz)来代替吗?

4

1 回答 1

11

什么时候datetime.now(pytz_timezone)失败?

据我所知,没有可能失败的情况。 在参数中传递的实例上datetime.now调用fromutc函数。tzinfo从 UTC 到本地时间的所有转换都是明确的,因此没有失败的机会。

此外,原始代码甚至不起作用。

d = est.normalize(EST)

这似乎将一个字符串作为唯一参数传递给normalize,该参数旨在采用datetime. 这给出了:

AttributeError: 'str' object has no attribute 'tzinfo'

我相信他们打算写:

d = est.normalize(d.astimezone(est))

也就是说,我不认为他们代码的冗长增加了太多价值。正如您所指出的,只需一步即可轻松完成:

d = datetime.now(est)

查看 的cpython 源代码datetime.now,我可以看到当tzinfo提供一个对象时,它会调用该fromutc对象上的方法。

if (self != NULL && tz != Py_None) {
    /* Convert UTC to tzinfo's zone. */
    PyObject *temp = self;

    self = _PyObject_CallMethodId(tz, &PyId_fromutc, "O", self);
    Py_DECREF(temp);
}

然后,在 pytz 源代码中,我看到该fromutc方法的实现方式不同,具体取决于区域是pytz.UTC,还是 的实例StaticTzInfo,还是DstTzInfo。在所有三种情况下,从输入 UTC 值到目标时区的转换是明确的。这是DstTzInfo实现,它是三个中更复杂的一个:

def fromutc(self, dt):
    '''See datetime.tzinfo.fromutc'''
    if (dt.tzinfo is not None
        and getattr(dt.tzinfo, '_tzinfos', None) is not self._tzinfos):
        raise ValueError('fromutc: dt.tzinfo is not self')
    dt = dt.replace(tzinfo=None)
    idx = max(0, bisect_right(self._utc_transition_times, dt) - 1)
    inf = self._transition_info[idx]
    return (dt + inf[0]).replace(tzinfo=self._tzinfos[inf])

这似乎会找到_utc_transition_times时区的转换,然后将其应用于返回的datetime. 这个方向没有歧义,所以结果是等价的。

同样值得注意的是,在文档datetime它说这datetime.now相当于调用:

tz.fromutc(datetime.utcnow().replace(tzinfo=tz))

鉴于fromutc我之前展示的 in pytz 的来源,我不确定这与以下内容有何不同:

tz.fromutc(datetime.utcnow())

localize但无论哪种情况,我都认为normalize没有必要。

于 2015-08-11T01:16:56.897 回答