我试图从datetime.datetime.today()的值中减去一个日期值,以计算多长时间以前的事情。但它抱怨道:
TypeError: can't subtract offset-naive and offset-aware datetimes
datetime.datetime.today()的返回值似乎不是“时区感知”,而我的其他日期值是。我如何从datetime.datetime.today()得到一个时区感知的返回值?
理想的解决方案是让它自动知道时区。
现在,它给我的时间是当地时间,恰好是PST,即UTC - 8小时。最坏的情况下,是否有一种方法我可以手动输入一个时区值到datetime.datetime.today()返回的datetime对象,并将其设置为UTC-8?
在标准库中,如果不创建自己的时区类,就没有跨平台的方法来创建感知时区。(编辑:Python 3.9在提供此功能的标准库中引入了zoneinfo。)
在Windows上,有win32timezone.utcnow(),但这是pywin32的一部分。我宁愿建议使用pytz库,它有一个不断更新的大多数时区的数据库。
使用本地时区可能非常棘手(请参阅下面的“进一步阅读”链接),因此您可能更希望在整个应用程序中使用UTC,特别是用于计算两个时间点之间的差值这样的算术操作。
你可以像这样得到当前的日期/时间:
import pytz
from datetime import datetime
datetime.utcnow().replace(tzinfo=pytz.utc)
注意,datetime.today()和datetime.now()返回本地时间,而不是UTC时间,因此对它们应用.replace(tzinfo=pytz.utc)是不正确的。
另一种很好的方法是:
datetime.now(pytz.utc)
它更短一点,做同样的事情。
进一步阅读/观看为什么在许多情况下更喜欢UTC:
pytz文档
关于时间,每个开发人员都应该知道的——许多现实生活用例的开发提示
时间和时区的问题-电脑爱好者-关于使用时区的复杂性的有趣的,令人大开眼界的解释(视频)
需要强调的是,从Python 3.6开始,你只需要标准库来获得一个表示本地时间的时区感知datetime对象(操作系统的设置)。使用astimezone ()
import datetime
datetime.datetime(2010, 12, 25, 10, 59).astimezone()
# e.g.
# datetime.datetime(2010, 12, 25, 10, 59, tzinfo=datetime.timezone(datetime.timedelta(seconds=3600), 'Mitteleuropäische Zeit'))
datetime.datetime(2010, 12, 25, 12, 59).astimezone().isoformat()
# e.g.
# '2010-12-25T12:59:00+01:00'
# I'm on CET/CEST
(见@johnchen902的评论)。
注意,这里有一个小警告,不要期望timedelta时区有任何“dst感知”。