Перетворити рік / місяць / день у день року на Python


127

Я використовую Python DATETIME модуль, а саме:

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

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

У будь-якому разі я бачу два варіанти:

A. Створіть число_of_days_in_month масив = [31, 28, ...], вирішіть, чи це високосний рік, підсумуйте дні вручну.

B. Використовуйте datetime.timedeltaдля здогадки та двійкового пошуку правильного дня року:

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

Обидва вони відчувають себе досить незграбно, і я відчуваю, що існує "пітонічний" спосіб розрахунку дня року. Якісь ідеї / пропозиції?

Відповіді:


255

Є дуже просте рішення:

from datetime import datetime
day_of_year = datetime.now().timetuple().tm_yday

13
Дуже незначне і, певно, педантичне доповнення, але використання, date.today()а не datetime.now()також, працює і ще більше підкреслює характер операції.
Джеремі

5
Ще краще: time.localtime().tm_yday Не потрібно перетворювати час дати в тимчасовий, оскільки саме це дає Localtime ().
Майк Елліс

14
@MikeEllis Але це рішення ілюструє, що робити, коли час не відповідає сьогоднішньому.
Герріт

2
Примітка: це починається відлік з 1
Мехді Неллен

1
Що робити, якщо я хочу зробити зворотний бік, у мене є число, яке дозволяє сказати "26 день року 2020", і я хочу, щоб він перетворився на дату "26-01-2020"?
Санкар

43

Ви не можете використовувати strftime?

>>> import datetime
>>> today = datetime.datetime.now()
>>> print today
2009-03-06 15:37:02.484000
>>> today.strftime('%j')
'065'

Редагувати

Як зазначається в коментарях, якщо ви хочете порівняти чи обчислити з цим числом, вам доведеться перетворити його, int()оскільки strftime()повертає рядок. Якщо це так, вам краще скористатися відповіддю DzinX .


1
-1: тьмяний. Краще алгоритм "сьогодні мінус 1 січня". Набагато чіткіше і абсолютно очевидно, не шукаючи деталей про "% j" стрілфа.
С.Лотт

12
Серйозно? Як це хиткий і відмітний 1 січня? STRELL існує з причини. Я просто не згоден. На мою думку, це ПОШИВЧИЙ очисник.
Паоло Бергантіно

1
Використання стропінгу непряме, оскільки воно створює рядок з числа: член timetuple.tm_yday. Прочитайте джерело. Створений рядок слід перетворити на число перед будь-якими розрахунками / порівняннями, так навіщо турбуватися?
tzot

2
Якщо вам потрібен день року в рядку, скажімо, для використання в імені файлу, то strroll - це набагато більш чисте рішення.
капітан_M

13

Відповідь 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

Я не впевнений у етикеті тут, але я вважав, що вказівник на зворотну функцію може бути корисним для інших, як я.


6

Я хочу представити продуктивність різних підходів на 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

1
Як ви порахували ефективність? Я спробував використовувати time.time, і це мої результати: def f (): (сьогодні - datetime.datetime (сьогодні. Рік, 1, 1)). Днів + 1 старт = time.time (); f (); print ('Lapsed {}'. format (time.time () - start)); Пропущено 5.221366882324219e-05 def f (): int (сьогодні.стріл ('% j')); start = time.time (); f (); print ('Lapsed {}'. format (time.time () - start)); Прострочено 7.462501525878906e-05
ссото

5

Просто відніміть 1 січня від дати:

import datetime
today = datetime.datetime.now()
day_of_year = (today - datetime.datetime(today.year, 1, 1)).days + 1

1
Нічого проти Паоло, але datetime.timedelta позначається в днях (а також секундах і мікросекундах, звичайно), тож це природний вибір --- звичайно більш прямий, ніж форматування рядків
zweiterlinde

1
d.toordinal () - дата (d.year, 1, 1) .toordinal () + 1 є більш стандартною згідно з документацією. Це еквівалент прийнятої відповіді, не створюючи цілий час кортежу.
Дінгл

5

Якщо у вас є підстави уникати використання 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

1

Цей код приймає дату як вхідний аргумент і повертає день року.

from datetime import datetime
date = raw_input("Enter date: ")  ## format is 02-02-2016
adate = datetime.datetime.strptime(date,"%d-%m-%Y")
day_of_year = adate.timetuple().tm_yday
day_of_year
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.