使用 strptime 将带有偏移量的时间戳转换为日期时间对象
- 2024-12-17 08:30:00
- admin 原创
- 147
问题描述:
我正在尝试使用 strptime 方法将格式为“2012-07-24T23:14:29-07:00”的时间戳转换为 Python 中的 datetime 对象。问题在于末尾的时间偏移量(-07:00)。如果没有偏移量,我可以成功完成
time_str = "2012-07-24T23:14:29"
time_obj=datetime.datetime.strptime(time_str,'%Y-%m-%dT%H:%M:%S')
但是我尝试了偏移量
time_str = "2012-07-24T23:14:29-07:00"
time_obj=datetime.datetime.strptime(time_str,'%Y-%m-%dT%H:%M:%S-%z').
但它给出了一个值错误,说“z”是一个错误的指令。
有什么解决方法吗?
解决方案 1:
Python 2strptime()
函数确实不支持%z
时区格式(因为底层time.strptime()
函数不支持)。您有两个选择:
解析时忽略时区
strptime
:
time_obj = datetime.datetime.strptime(time_str[:19], '%Y-%m-%dT%H:%M:%S')
使用该
dateutil
模块,它的解析函数确实处理时区:
from dateutil.parser import parse
time_obj = parse(time_str)
命令提示符上的快速演示:
>>> from dateutil.parser import parse
>>> parse("2012-07-24T23:14:29-07:00")
datetime.datetime(2012, 7, 24, 23, 14, 29, tzinfo=tzoffset(None, -25200))
您还可以升级到 Python 3.2 或更新版本,其中时区支持已经得到改进,可以正常工作,只要您从输入中%z
删除最后一个,并删除之前的::
`-`%z
>>> import datetime
>>> time_str = "2012-07-24T23:14:29-07:00"
>>> datetime.datetime.strptime(time_str, '%Y-%m-%dT%H:%M:%S%z')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python3.4/_strptime.py", line 500, in _strptime_datetime
tt, fraction = _strptime(data_string, format)
File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python3.4/_strptime.py", line 337, in _strptime
(data_string, format))
ValueError: time data '2012-07-24T23:14:29-07:00' does not match format '%Y-%m-%dT%H:%M:%S%z'
>>> ''.join(time_str.rsplit(':', 1))
'2012-07-24T23:14:29-0700'
>>> datetime.datetime.strptime(''.join(time_str.rsplit(':', 1)), '%Y-%m-%dT%H:%M:%S%z')
datetime.datetime(2012, 7, 24, 23, 14, 29, tzinfo=datetime.timezone(datetime.timedelta(-1, 61200)))
解决方案 2:
在 Python 3.7+ 中:
from datetime import datetime
time_str = "2012-07-24T23:14:29-07:00"
dt_aware = datetime.fromisoformat(time_str)
print(dt_aware.isoformat('T'))
# -> 2012-07-24T23:14:29-07:00
在 Python 3.2+ 中:
from datetime import datetime
time_str = "2012-07-24T23:14:29-0700"
dt_aware = datetime.strptime(time_str, '%Y-%m-%dT%H:%M:%S%z')
print(dt_aware.isoformat('T'))
# -> 2012-07-24T23:14:29-07:00
注意:在 Python 3.7 之前,此变体不支持:
部分内容( rfc 3339-0700
允许这两种格式)。请参阅datetime:添加解析 RFC 3339 日期和时间的能力。
在较旧的 Python 版本(例如 Python 2.7)上,您可以手动解析 utc 偏移量:
from datetime import datetime
time_str = "2012-07-24T23:14:29-0700"
# split the utc offset part
naive_time_str, offset_str = time_str[:-5], time_str[-5:]
# parse the naive date/time part
naive_dt = datetime.strptime(naive_time_str, '%Y-%m-%dT%H:%M:%S')
# parse the utc offset
offset = int(offset_str[-4:-2])*60 + int(offset_str[-2:])
if offset_str[0] == "-":
offset = -offset
dt = naive_dt.replace(tzinfo=FixedOffset(offset))
print(dt.isoformat('T'))
其中FixedOffset
类在这里定义。
解决方案 3:
ValueError: 'z' is a bad directive in format...
(注意:在我的例子中我必须坚持使用 Python 2.7)
我在解析输出中的提交日期时遇到了类似的问题,其实际上不是 ISO8601 格式(因此在更高版本中git log --date=iso8601
添加了)。--date=iso8601-strict
因为我正在使用,所以django
我可以利用那里的实用程序。
https://github.com/django/django/blob/master/django/utils/dateparse.py
>>> from django.utils.dateparse import parse_datetime
>>> parse_datetime('2013-07-23T15:10:59.342107+01:00')
datetime.datetime(2013, 7, 23, 15, 10, 59, 342107, tzinfo=+0100)
您也strptime
可以使用自己的正则表达式。
解决方案 4:
用python 3.5.2
来转换26 Sep 2000 05:11:00 -0700
from datetime import datetime
dt_obj = datetime.strptime("26 Sep 2000 05:11:00 -0700", '%d %b %Y %H:%M:%S %z')
转换2012-07-24T23:14:29 -0700
dt_obj = datetime.strptime('2012-07-24T23:14:29 -0700', '%Y-%m-%dT%H:%M:%S %z')
Python 3.5.2 不支持 -07:00 时间偏移 ':' 应该被删除