Преобразование года / месяца / дня в день года в Python

Я использую модуль datetime, то есть:

>>> import datetime
>>> today = datetime.datetime.now()
>>> print(today)
2009-03-06 13:24:58.857946

и я хотел бы вычислить день года с учетом високосных лет. например сегодня (6 марта 2009 года) 65-й день 2009 года.

Я вижу два варианта:

  1. Создайте массив number_of_days_in_month = [31, 28, ...], решите, является ли это високосным годом, и вручную просуммируйте дни.

  2. Используйте datetime.timedelta, чтобы угадать, а затем выполните двоичный поиск правильного дня года:

    >>> import datetime
    >>> YEAR = 2009
    >>> DAY_OF_YEAR = 62
    >>> d = datetime.date(YEAR, 1, 1) + datetime.timedelta(DAY_OF_YEAR - 1)
    

Они оба кажутся довольно неуклюжими, и у меня есть интуиция, что есть более питонический способ вычисления дня в году. Есть идеи / предложения?


person Pete    schedule 06.03.2009    source источник


Ответы (6)


Используйте datetime.timetuple(), чтобы преобразовать ваш datetime объект в _ 3_ объект, затем получите его _ 4_:

from datetime import datetime
day_of_year = datetime.now().timetuple().tm_yday  # returns 1 for January 1st
person DzinX    schedule 08.03.2009
comment
Очень незначительное и, возможно, педантичное дополнение, но использование date.today() вместо datetime.now() также работает и немного больше подчеркивает характер операции. - person Jeremy; 30.12.2013
comment
примечание: отсчет начинается с 1 - person Mehdi Nellen; 21.11.2019
comment
что, если я хочу сделать обратное, у меня есть число, скажем, 26-й день 2020 года, и я хочу преобразовать его в дату 26-01-2020? - person Sankar; 26.01.2020
comment
@Sankar, тогда вы можете найти этот вопрос в Google и задать свой собственный вопрос о переполнении стека, если вы ничего не найдете. - person Boris; 01.11.2020

Вы можете использовать strftime с _ 2_ строка формата:

>>> import datetime
>>> today = datetime.datetime.now()
>>> today.strftime('%j')
'065'

но если вы хотите провести сравнение или вычисления с этим числом, вам придется преобразовать его в int(), потому что strftime() возвращает строку. В этом случае лучше использовать ответ DzinX.

person Paolo Bergantino    schedule 06.03.2009
comment
Использование strftime является косвенным, потому что оно создает строку из числа: члена timetuple.tm_yday. Прочтите источник. Созданная строка должна быть преобразована в число перед любыми вычислениями / сравнениями, так зачем беспокоиться? - person tzot; 08.03.2009
comment
Если кому-то нужен день года в строке, скажем, для использования в имени файла, то strftime - гораздо более чистое решение. - person captain_M; 15.11.2014

Ответ DZinX - отличный ответ на этот вопрос. Я нашел этот вопрос и использовал ответ DZinX, ища обратную функцию: преобразовать даты с юлианским днем ​​года в дата и время.

Я обнаружил, что это работает:

import datetime
datetime.datetime.strptime('1936-077T13:14:15','%Y-%jT%H:%M:%S')

>>>> datetime.datetime(1936, 3, 17, 13, 14, 15)

datetime.datetime.strptime('1936-077T13:14:15','%Y-%jT%H:%M:%S').timetuple().tm_yday

>>>> 77

Или численно:

import datetime
year,julian = [1936,77]
datetime.datetime(year, 1, 1)+datetime.timedelta(days=julian -1)

>>>> datetime.datetime(1936, 3, 17, 0, 0)

Или с jdates, основанными на дробной единице, популярными в некоторых доменах:

jdate_frac = (datetime.datetime(1936, 3, 17, 13, 14, 15)-datetime.datetime(1936, 1, 1)).total_seconds()/86400+1
display(jdate_frac)

>>>> 77.5515625

year,julian = [1936,jdate_frac]
display(datetime.datetime(year, 1, 1)+datetime.timedelta(days=julian -1))

>>>> datetime.datetime(1936, 3, 17, 13, 14, 15)

Я не уверен в соблюдении этикета здесь, но я подумал, что указатель на обратную функциональность может быть полезен для других, таких как я.

person Dave X    schedule 23.10.2012

Если у вас есть причина избегать использования модуля datetime, эти функции будут работать.

def is_leap_year(year):
    """ if year is a leap year return True
        else return False """
    if year % 100 == 0:
        return year % 400 == 0
    return year % 4 == 0

def doy(Y,M,D):
    """ given year, month, day return day of year
        Astronomical Algorithms, Jean Meeus, 2d ed, 1998, chap 7 """
    if is_leap_year(Y):
        K = 1
    else:
        K = 2
    N = int((275 * M) / 9.0) - K * int((M + 9) / 12.0) + D - 30
    return N

def ymd(Y,N):
    """ given year = Y and day of year = N, return year, month, day
        Astronomical Algorithms, Jean Meeus, 2d ed, 1998, chap 7 """    
    if is_leap_year(Y):
        K = 1
    else:
        K = 2
    M = int((9 * (K + N)) / 275.0 + 0.98)
    if N < 32:
        M = 1
    D = N - int((275 * M) / 9.0) + K * int((M + 9) / 12.0) + 30
    return Y, M, D
person Barry Andersen    schedule 11.05.2015
comment
Вы также можете просто скопировать / вставить исходный код datetime github .com / python / cpython / blob /, но зачем вам это, если вы можете просто import datetime? - person Boris; 01.11.2020

Просто вычтите 1 января из даты:

import datetime
today = datetime.datetime.now()
day_of_year = (today - datetime.datetime(today.year, 1, 1)).days + 1
person zweiterlinde    schedule 06.03.2009
comment
d.toordinal () - date (d.year, 1, 1) .toordinal () + 1 является более стандартным согласно документации. Это эквивалентно принятому ответу без создания всего временного кортежа. - person Dingle; 30.04.2010

Я хочу представить производительность различных подходов на Python 3.4, Linux x64. Выдержка из профилировщика строк:

      Line #      Hits         Time  Per Hit   % Time  Line Contents
      ==============================================================
         (...)
         823      1508        11334      7.5     41.6          yday = int(period_end.strftime('%j'))
         824      1508         2492      1.7      9.1          yday = period_end.toordinal() - date(period_end.year, 1, 1).toordinal() + 1
         825      1508         1852      1.2      6.8          yday = (period_end - date(period_end.year, 1, 1)).days + 1
         826      1508         5078      3.4     18.6          yday = period_end.timetuple().tm_yday
         (...)

Таким образом, наиболее эффективным является

yday = (period_end - date(period_end.year, 1, 1)).days
person omikron    schedule 15.09.2014
comment
Как вы рассчитывали производительность? Я пробовал использовать time.time и вот мои результаты: def f (): (today - datetime.datetime (today.year, 1, 1)). Days + 1 start = time.time (); f (); print ('Lapsed {}'. format (time.time () - start)); Истекло 5.221366882324219e-05 def f (): int (today.strftime ('% j')); начало = время.время (); f (); print ('Lapsed {}'. format (time.time () - start)); По истечении срока службы 7.462501525878906e-05 - person ssoto; 22.08.2017