为什么python 2.7在UTC日期时间对象的isoformat字符串末尾不包含Z字符(Zulu或零偏移量)?与JavaScript不同?
>>> datetime.datetime.utcnow().isoformat()
'2013-10-29T09:14:03.895210'
而在javascript中
>>> console.log(new Date().toISOString());
2013-10-29T09:38:41.341Z
答案 0 :(得分:37)
Python datetime
对象默认没有时区信息,没有它,Python实际上违反了ISO 8601规范(if no time zone info is given, assumed to be local time)。您可以使用pytz package获取一些默认时区,或直接自己子类tzinfo
:
from datetime import datetime, tzinfo, timedelta
class simple_utc(tzinfo):
def tzname(self,**kwargs):
return "UTC"
def utcoffset(self, dt):
return timedelta(0)
然后,您可以手动将时区信息添加到utcnow()
:
>>> datetime.utcnow().replace(tzinfo=simple_utc()).isoformat()
'2014-05-16T22:51:53.015001+00:00'
请注意,此DOES符合ISO 8601格式,允许Z
或+00:00
作为UTC的后缀。请注意,后者实际上更符合标准,一般如何表示时区(UTC是一种特殊情况。)
答案 1 :(得分:32)
isoformat()
Python的datetime
不支持military timezone后缀,例如UTC的'Z'后缀。以下简单的字符串替换可以解决这个问题:
In [1]: import datetime
In [2]: d = datetime.datetime(2014, 12, 10, 12, 0, 0)
In [3]: str(d).replace('+00:00', 'Z')
Out[3]: '2014-12-10 12:00:00Z'
str(d)
与d.isoformat(sep=' ')
请参阅:Datetime, Python Standard Library
strftime()
或者您可以使用strftime
来达到同样的效果:
In [4]: d.strftime('%Y-%m-%d %H:%M:%SZ')
Out[4]: '2014-12-10 12:00:00Z'
注意:仅当您知道指定的日期是UTC时,此选项才有效。
更进一步,您可能有兴趣使用pytz
strftime
时区标记%Z
显示人类可读的时区信息:
In [5]: import pytz
In [6]: d = datetime.datetime(2014, 12, 10, 12, 0, 0, tzinfo=pytz.utc)
In [7]: d
Out[7]: datetime.datetime(2014, 12, 10, 12, 0, tzinfo=<UTC>)
In [8]: d.strftime('%Y-%m-%d %H:%M:%S %Z')
Out[8]: '2014-12-10 12:00:00 UTC'
答案 2 :(得分:7)
将UTC时区对象传递给datetime.now()
而不是使用datetime.utcnow()
:
from datetime import datetime, timezone
datetime.now(timezone.utc)
>>> datetime.datetime(2020, 1, 8, 6, 6, 24, 260810, tzinfo=datetime.timezone.utc)
datetime.now(timezone.utc).isoformat()
>>> '2020-01-08T06:07:04.492045+00:00'
看起来 很好,让我们看看Django和dateutil
的想法:
from django.utils.timezone import is_aware
is_aware(datetime.now(timezone.utc))
>>> True
from dateutil.parser import isoparse
is_aware(isoparse(datetime.now(timezone.utc).isoformat()))
>>> True
请注意,您需要使用isoparse()
,因为datetime.fromisoformat()
的文档说它“不支持解析任意ISO 8601字符串”。
好的,Python日期时间对象和ISO 8601字符串是UTC“可识别的”。现在,让我们看一下JavaScript对datetime字符串的看法。从this answer借用,我们得到:
let date= '2020-01-08T06:07:04.492045+00:00';
const dateParsed = new Date(Date.parse(date))
document.write(dateParsed);
document.write("\n");
// Tue Jan 07 2020 22:07:04 GMT-0800 (Pacific Standard Time)
document.write(dateParsed.toISOString());
document.write("\n");
// 2020-01-08T06:07:04.492Z
document.write(dateParsed.toUTCString());
document.write("\n");
// Wed, 08 Jan 2020 06:07:04 GMT
注释:
我通过几个目标解决了这个问题:
timezone
实用程序功能和dateutil
解析器验证datetime对象和字符串请注意,此方法不包括Z后缀,并且不使用utcnow()
。但是它基于recommendation in the Python documentation,并且通过Django和JavaScript传递了摘要。
您可能还想阅读this blog post。
答案 3 :(得分:6)
Python日期时间有点笨重。使用arrow
。
> str(arrow.utcnow())
'2014-05-17T01:18:47.944126+00:00'
Arrow与datetime基本上具有相同的api,但是时区和一些额外的细节应该在主库中。
可以通过以下方式实现与Javascript兼容的格式:
arrow.utcnow().isoformat().replace("+00:00", "Z")
'2018-11-30T02:46:40.714281Z'
Javascript Date.parse
将从时间戳中悄悄地减少微秒。
答案 4 :(得分:4)
datetime.now(timezone.utc).isoformat().replace("+00:00", "Z")
之所以不包含“ Z”,是因为datetime.now()
甚至datetime.utcnow()
返回时区的原始日期时间,即没有关联时区信息的日期时间。要获取时区感知的日期时间,您需要将时区作为参数传递给datetime now
。例如:
from datetime import datetime, timezone
datetime.utcnow()
#> datetime.datetime(2020, 9, 3, 20, 58, 49, 22253)
# This is timezone naive
datetime.now(timezone.utc)
#> datetime.datetime(2020, 9, 3, 20, 58, 49, 22253, tzinfo=datetime.timezone.utc)
# This is timezone aware
一旦有了时区感知时间戳,isoformat就会包含一个时区名称。因此,您可以通过以下方式获取ISO 8601时间戳:
datetime.now(timezone.utc).isoformat()
#> '2020-09-03T20:53:07.337670+00:00'
“ + 00:00”是UTC的有效ISO 8601时区名称。如果要用“ Z”代替“ +00:00”,则必须自己进行替换:
datetime.now(timezone.utc).isoformat().replace("+00:00", "Z")
#> '2020-09-03T20:53:07.337670Z'
答案 5 :(得分:3)
以下javascript和python脚本提供相同的输出。我认为这就是您要寻找的。 p>
JavaScript
new Date().toISOString()
Python
import datetime
datetime.datetime.utcnow().strftime('%Y-%m-%dT%H:%M:%S.%f')[:-3] + 'Z'
他们提供的输出是utc(zelda)时间,格式为ISO字符串,有效数字为3毫秒,并附加Z。
2019-01-19T23:20:25.459Z
答案 6 :(得分:2)
在Python> = 3.2中,您可以简单地使用以下代码:
>>> from datetime import datetime, timezone
>>> datetime.now(timezone.utc).isoformat()
'2019-03-14T07:55:36.979511+00:00'
答案 7 :(得分:1)
通过结合上面的所有答案,我得到了以下功能:
from datetime import datetime, tzinfo, timedelta
class simple_utc(tzinfo):
def tzname(self,**kwargs):
return "UTC"
def utcoffset(self, dt):
return timedelta(0)
def getdata(yy, mm, dd, h, m, s) :
d = datetime(yy, mm, dd, h, m, s)
d = d.replace(tzinfo=simple_utc()).isoformat()
d = str(d).replace('+00:00', 'Z')
return d
print getdata(2018, 02, 03, 15, 0, 14)
答案 8 :(得分:1)
帖子上有很多不错的答案,但我希望格式能完全像JavaScript一样。这就是我正在使用的并且效果很好。
In [1]: import datetime
In [1]: now = datetime.datetime.utcnow()
In [1]: now.strftime('%Y-%m-%dT%H:%M:%S') + now.strftime('.%f')[:4] + 'Z'
Out[3]: '2018-10-16T13:18:34.856Z'
答案 9 :(得分:1)
pip install python-dateutil
>>> a = "2019-06-27T02:14:49.443814497Z"
>>> dateutil.parser.parse(a)
datetime.datetime(2019, 6, 27, 2, 14, 49, 443814, tzinfo=tzutc())
答案 10 :(得分:0)
我使用摆锤:
import pendulum
d = pendulum.now("UTC").to_iso8601_string()
print(d)
>>> 2019-10-30T00:11:21.818265Z
答案 11 :(得分:-2)
>>> import arrow
>>> now = arrow.utcnow().format('YYYY-MM-DDTHH:mm:ss.SSS')
>>> now
'2018-11-28T21:34:59.235'
>>> zulu = "{}Z".format(now)
>>> zulu
'2018-11-28T21:34:59.235Z'
或者,一口气获得它:
>>> zulu = "{}Z".format(arrow.utcnow().format('YYYY-MM-DDTHH:mm:ss.SSS'))
>>> zulu
'2018-11-28T21:54:49.639Z'