【发布时间】:2011-01-20 20:32:07
【问题描述】:
datetime.datetime.utcnow()
为什么datetime 没有任何时区信息,因为它明确是 UTC datetime?
我希望这将包含tzinfo。
【问题讨论】:
-
如何将字符串类型的普通iso格式日期字段转换为utc格式?
datetime.datetime.utcnow()
为什么datetime 没有任何时区信息,因为它明确是 UTC datetime?
我希望这将包含tzinfo。
【问题讨论】:
UTC 日期不需要任何时区信息,因为它们是 UTC,根据定义,这意味着它们没有偏移量。
【讨论】:
pytz.utc)。请注意,与 UTC 的偏移量未知的值与已知为 0 的值之间存在很大差异。后者是 utcnow() 应该 返回的值,IMO。根据文档,这将符合“一个有意识的对象用于表示一个不可解释的特定时刻”。
标准 Python 库在 Python 3.2 之前不包含任何 tzinfo 类。我只能猜测原因。就我个人而言,我认为不包含 UTC 的 tzinfo 类是错误的,因为该类没有争议,足以拥有标准实现。虽然库中没有实现,但在tzinfo documentation 中给出了一个示例。
from datetime import timedelta, tzinfo
ZERO = timedelta(0)
# A UTC class.
class UTC(tzinfo):
"""UTC"""
def utcoffset(self, dt):
return ZERO
def tzname(self, dt):
return "UTC"
def dst(self, dt):
return ZERO
utc = UTC()
一旦你有了一个 UTC tzinfo 对象,你仍然不能将它与 utcnow 一起使用。获取当前时间作为感知日期时间对象:
from datetime import datetime
now = datetime.now(utc)
在 Python 3.2 中,他们最终将 UTC tzinfo 类放入库中:
from datetime import datetime, timezone
now = datetime.now(timezone.utc)
在 Python 3.9 中,他们为所有其他时区创建了 tzinfo 类。详情请见PEP 615 -- Support for the IANA Time Zone Database in the Standard Library。
【讨论】:
utcnow()创建的datetime对象)...
struct 模块会自动将 Unicode 转换为字节串,最终决定打破与早期 Python 3 版本的兼容性,以防止做出错误的决定。
tzinfo 文档包含实现它的代码示例,但它们并没有在 datetime 本身中包含该功能! docs.python.org/2/library/datetime.html#datetime.tzinfo.fromutc
pytz 模块的实现所包含的内容。它添加了一些有用的方法。
pytz 是一个很好的资源。当我编辑我的答案以放入示例代码时,其他人已经提出了建议,我不想抢走他们的风头。
这意味着它是天真的时区,所以你不能将它与datetime.astimezone一起使用
你可以给它一个这样的时区
import pytz # 3rd party: $ pip install pytz
u = datetime.utcnow()
u = u.replace(tzinfo=pytz.utc) #NOTE: it works only with a fixed utc offset
现在您可以更改时区
print(u.astimezone(pytz.timezone("America/New_York")))
要获取给定时区的当前时间,您可以将 tzinfo 直接传递给datetime.now():
#!/usr/bin/env python
from datetime import datetime
import pytz # $ pip install pytz
print(datetime.now(pytz.timezone("America/New_York")))
它适用于任何时区,包括那些遵守夏令时 (DST) 的时区,即它适用于在不同时间可能具有不同 utc 偏移量的时区(非固定 utc 偏移量)。不要使用tz.localize(datetime.now())——当本地时间不明确时,它可能会在夏令时结束过渡期间失败。
【讨论】:
astimezone 中的某些城市时区,可能会出错。所以 datetime 不仅没有本地时区,而且 tzinfo 唯一广泛可用的实现不符合假定的标准。
u=datetime.now(pytz.utc)怎么样
tz.localize(datetime.now());请改用datetime.now(tz)。
pytz 模块是一个选项,还有另一个 python-dateutil,虽然也是第三方包,但可能已经可用,具体取决于您的其他依赖项和操作系统。
我只是想包含此方法以供参考-如果您已经为其他目的安装了python-dateutil,则可以使用它的tzinfo 而不是与pytz 复制
import datetime
import dateutil.tz
# Get the UTC time with datetime.now:
utcdt = datetime.datetime.now(dateutil.tz.tzutc())
# Get the UTC time with datetime.utcnow:
utcdt = datetime.datetime.utcnow()
utcdt = utcdt.replace(tzinfo=dateutil.tz.tzutc())
# For fun- get the local time
localdt = datetime.datetime.now(dateutil.tz.tzlocal())
我倾向于同意对utcnow 的调用应包含UTC 时区信息。我怀疑这不包括在内,因为本机日期时间库默认为天真的日期时间以实现交叉兼容性。
【讨论】:
utcdt = datetime.datetime.utcfromtimestamp(1234567890).replace(dateutil.tz.tzutc())
datetime.now(pytz_tz) 总是有效的; datetime.now(dateutil.tz.tzlocal()) may fail during DST transitions。 PEP 495 -- Local Time Disambiguation 将来可能会改善 dateutil 的情况。
utc_dt = datetime.fromtimestamp(1234567890, dateutil.tz.tzutc())(注意:dateutil with a non-fixed utc offset (such as dateutil.tz.tzlocal()) may fail here,使用a pytz-based solution instead)。
dateutil.parser 导入了dateutil,所以我最喜欢这个解决方案。它很简单:utcCurrentTime = datetime.datetime.now(tz=dateutil.tz.tzutc())。中提琴!!
请注意,对于 Python 3.2 及更高版本,datetime 模块包含 datetime.timezone。 datetime.utcnow() 的文档说:
可以通过调用
datetime.now(timezone.utc)来获取当前的UTC日期时间。
所以,datetime.utcnow() 没有设置 tzinfo 来表示它是 UTC,但 datetime.now(datetime.timezone.utc) 确实返回 UTC 时间 tzinfo 设置。
所以你可以这样做:
>>> import datetime
>>> datetime.datetime.now(datetime.timezone.utc)
datetime.datetime(2014, 7, 10, 2, 43, 55, 230107, tzinfo=datetime.timezone.utc)
【讨论】:
datetime.now(timezone.utc) 或 datetime.utcnow(timezone.utc)?
datetime.utcnow() 不接受任何参数。所以它必须是datetime.now(timezone.utc)。
datetime.now() 将返回机器时间,但 datetime.utcnow() 将返回实际 UTC 时间。
datetime.utcnow() 没有设置 tzinfo 来表示它是 UTC。但datetime.now(datetime.timezone.utc) 确实返回 UTC 时间 with tzinfo 设置。
from datetime import datetime
from dateutil.relativedelta import relativedelta
d = datetime.now()
date = datetime.isoformat(d).split('.')[0]
d_month = datetime.today() + relativedelta(months=1)
next_month = datetime.isoformat(d_month).split('.')[0]
【讨论】:
Julien Danjou 写了一篇很好的文章来解释why you should never deal with timezones。摘录:
确实,Python 日期时间 API 总是返回未知的日期时间对象, 这是非常不幸的。确实,一旦你得到其中之一 对象,没有办法知道时区是什么,因此这些 对象本身就相当“无用”。
唉,即使您可以使用utcnow(),您仍然不会看到您发现的时区信息。
建议:
始终使用感知
datetime对象,即带有时区信息。那 确保您可以直接比较它们(知道和不知道datetime对象不可比较)并将它们正确返回给用户。 利用 pytz 拥有时区对象。使用 ISO 8601 作为输入和 输出字符串格式。使用
datetime.datetime.isoformat()返回 时间戳作为使用该格式格式化的字符串,其中包括 时区信息。如果您需要解析包含 ISO 8601 格式时间戳的字符串, 你可以依赖
iso8601,它返回正确的时间戳 时区信息。这使得时间戳可以直接比较。
【讨论】:
在 Python 3.2+ 中添加timezone 信息
import datetime
>>> d = datetime.datetime.now(tz=datetime.timezone.utc)
>>> print(d.tzinfo)
'UTC+00:00'
【讨论】:
AttributeError: 'module' object has no attribute 'timezone' Python 2.7.13(默认,2017 年 1 月 19 日,14:48:08)
from datetime import datetime, timezone 然后调用datetime.now(tz=timezone.utc)
datetime.datetime.utcnow() 将 UTC 时间作为原始日期时间对象返回的行为显然是有问题的,必须加以修复。如果您的系统本地时区不是 UTC,它可能会导致意外结果,因为 datetime 库假定天真的 datetime 对象来表示系统本地时间。例如,datetime.datetime.utcnow().timestaamp() 给出了我计算机上正确值前 4 小时的时间戳。此外,从 python 3.6 开始,datetime.astimezone() 可以在幼稚的日期时间实例上调用,但 datetime.datetime.utcnow().astimezone(any_timezone) 会给出错误的结果,除非您的系统本地时区是 UTC。
【讨论】: