在Python中生成RFC 3339时间戳

Yar*_*rin 51 python datetime iso8601 rfc3339

我正在尝试在Python中生成RFC 3339 UTC时间戳.到目前为止,我已经能够做到以下几点:

>>> d = datetime.datetime.now()
>>> print d.isoformat('T')
2011-12-18T20:46:00.392227
Run Code Online (Sandbox Code Playgroud)

我的问题是设置UTC偏移量.

根据该文件,在类方法datetime.now([tz]),有一个可选的tz参数,其中tz must be an instance of a class tzinfo subclass,和datetime.tzinfoan abstract base class for time zone information objects.

这就是我迷路的地方 - 为什么tzinfo是一个抽象类,我该如何实现呢?


(注意:在PHP中它很简单timestamp = date(DATE_RFC3339);,这就是为什么我无法理解为什么Python的方法如此错综复杂......)

mon*_*kut 41

时区是一种痛苦,这可能是他们选择不将它们包含在日期时间库中的原因.

尝试pytz,它有你想要的tzinfo:http://pytz.sourceforge.net/

您需要先创建datetime对象,然后按如下所示应用时区,然后.isoformat()输出将根据需要包含UTC偏移:

d = datetime.datetime.utcnow()
d_with_timezone = d.replace(tzinfo=pytz.UTC)
d_with_timezone.isoformat()
Run Code Online (Sandbox Code Playgroud)

'2017-04-13T14:34:23.111142 + 00:00'

或者,只使用UTC,并在末尾抛出"Z"(对于Zulu时区)以将"时区"标记为UTC.

d = datetime.datetime.utcnow() # <-- get time in UTC
print d.isoformat("T") + "Z"
Run Code Online (Sandbox Code Playgroud)

'2017-04-13T14:34:23.111142Z'


jfs*_*jfs 26

在Python 3.3+中:

>>> from datetime import datetime, timezone                                
>>> local_time = datetime.now(timezone.utc).astimezone()
>>> local_time.isoformat()
'2015-01-16T16:52:58.547366+01:00'
Run Code Online (Sandbox Code Playgroud)

在较旧的Python版本中,如果您只需要一个表示当前UTC时间的日期时间对象,那么您可以定义一个简单的tzinfo子类,如文档中所示,以表示UTC时区:

from datetime import datetime

utc_now = datetime.now(utc)
print(utc_now.isoformat('T'))
# -> 2015-05-19T20:32:12.610841+00:00
Run Code Online (Sandbox Code Playgroud)

您还可以使用tzlocal模块获取pytz代表当地时区的时区:

#!/usr/bin/env python
from datetime import datetime
from tzlocal import get_localzone # $ pip install tzlocal

now = datetime.now(get_localzone())
print(now.isoformat('T'))
Run Code Online (Sandbox Code Playgroud)

它适用于Python 2和3.

  • @villapx:rfc 3339 是 ISO 8601 的一个配置文件。碰巧 `datetime.isoformat()` 是用于时区感知日期时间的 RFC 3339(这就是我使用它的原因)。 (2认同)

JJC*_*JJC 10

在现代 (3.x) python 上,要获得 RFC 3339 UTC 时间,您需要做的就是使用 datetime 和这一行(不需要第三方模块):

import datetime
datetime.datetime.now(datetime.timezone.utc).isoformat()
Run Code Online (Sandbox Code Playgroud)

结果类似于:'2019-06-13T15:29:28.972488+00:00'

此 ISO 8601 字符串也与 RFC3339 兼容。

  • @uwieuwe4 你可以在上面的结果上使用 `.replace('+00:00', 'Z')` 。或者,只需使用 `.strftime('%Y-%m-%dT%H:%M:%SZ')` 而不是 .isoformat()。 (4认同)

rua*_*akh 7

再往下,在您链接到同一文档,它解释了如何实现它,给一些例子,包括一个完整的代码UTC类(代表UTC),一FixedOffset类(代表与UTC的固定偏移,而不是一个时区时区与DST和诸如此类的东西)和其他一些.

  • 为什么不在问题中包含完整示例而不是链接,这是stackoverflow的首选方式?@gene_wood为什么不使用您的完整要点解决方案创建新答案? (2认同)

Men*_*cus 7

我经常使用RFC3339日期时间格式,但我找到了一个合适的解决方案,可以在两个方向上转换date_string <=> datetime_object.

您需要两个不同的外部模块,因为其中一个模块只能在一个方向上进行转换(不幸的是):

首次安装:

sudo pip install rfc3339
sudo pip install iso8601
Run Code Online (Sandbox Code Playgroud)

然后包括:

import datetime     # for general datetime object handling
import rfc3339      # for date object -> date string
import iso8601      # for date string -> date object
Run Code Online (Sandbox Code Playgroud)

为了不需要记住哪个模块是哪个方向,我写了两个简单的辅助函数:

def get_date_object(date_string):
  return iso8601.parse_date(date_string)

def get_date_string(date_object):
  return rfc3339.rfc3339(date_object)
Run Code Online (Sandbox Code Playgroud)

您可以在代码中轻松使用,如下所示:

input_string = '1989-01-01T00:18:07-05:00'
test_date = get_date_object(input_string)
# >>> datetime.datetime(1989, 1, 1, 0, 18, 7, tzinfo=<FixedOffset '-05:00' datetime.timedelta(-1, 68400)>)

test_string = get_date_string(test_date)
# >>> '1989-01-01T00:18:07-05:00'

test_string is input_string # >>> True
Run Code Online (Sandbox Code Playgroud)

Heureka!现在您可以轻松(哈哈)以可用的格式使用日期字符串和日期字符串.

  • “赫里卡”。失去'H'。 (2认同)