Вопрос-Ответ

Convert timestamps with offset to datetime obj using strptime

Преобразование временных меток со смещением в datetime obj с помощью strptime

Я пытаюсь преобразовать временные метки формата "2012-07-24T23: 14:29-07:00" в объекты datetime в python с помощью метода strptime. Проблема связана со смещением по времени в конце (-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

strptime() Функция Python 2 действительно не поддерживает %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 этот вариант не поддерживался : в -0700 части (оба формата разрешены rfc 3339). Смотрите 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)

У меня была аналогичная проблема при анализе дат фиксации из выходных данных git log --date=iso8601 которые на самом деле не являются форматом 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 ':' должно быть удалено

2024-01-29 22:16 python datetime