Как рассчитать количество дней между двумя заданными датами?

Если у меня две даты (например, '8/18/2008' и '9/26/2008'), как лучше всего узнать количество дней между этими двумя датами?

Почему в Python есть оператор "pass"?
Почему в Python есть оператор "pass"?
Оператор pass в Python - это простая концепция, которую могут быстро освоить даже новички без опыта программирования.
Некоторые методы, о которых вы не знали, что они существуют в Python
Некоторые методы, о которых вы не знали, что они существуют в Python
Python - самый известный и самый простой в изучении язык в наши дни. Имея широкий спектр применения в области машинного обучения, Data Science,...
Основы Python Часть I
Основы Python Часть I
Вы когда-нибудь задумывались, почему в программах на Python вы видите приведенный ниже код?
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
Алиса и Боб имеют неориентированный граф из n узлов и трех типов ребер:
Оптимизация кода с помощью тернарного оператора Python
Оптимизация кода с помощью тернарного оператора Python
И последнее, что мы хотели бы показать вам, прежде чем двигаться дальше, это
Советы по эффективной веб-разработке с помощью Python
Советы по эффективной веб-разработке с помощью Python
Как веб-разработчик, Python может стать мощным инструментом для создания эффективных и масштабируемых веб-приложений.
571
0
628 773
13
Перейти к ответу Данный вопрос помечен как решенный

Ответы 13

Ответ принят как подходящий

Если у вас есть два объекта даты, вы можете просто вычесть их, что вычисляет объект timedelta.

from datetime import date

d0 = date(2008, 8, 18)
d1 = date(2008, 9, 26)
delta = d1 - d0
print(delta.days)

Соответствующий раздел документации: https://docs.python.org/library/datetime.html.

См. Другой пример в этот ответ.

Отличные ответы здесь. Поскольку многие люди могут использовать фрейм данных pandas, может быть полезно проверить ссылку о том, как преобразовать из np.datetime64 в python datetimestackoverflow.com/questions/52982056/…

Pramit 16.05.2019 01:54

Приятно то, что это решение также возвращает правильную дельту для високосных лет.

Lasma 02.05.2020 11:19

Вот почему я люблю питон - такие вещи, благодаря которым техническая сложная проблема кажется решенной естественным образом.

martyglaubitz 30.07.2020 12:40

Это учитывает дополнительные секунды?

run_the_race 23.10.2020 15:57

если вам нужно исключить выходные и праздничные дни (рабочие дни), попробуйте этот github.com/cadu-leite/networkdays - это просто и не добавляйте третьих зависимостей

Carlos Leite 29.12.2020 01:11

Используя силу datetime:

from datetime import datetime
date_format = "%m/%d/%Y"
a = datetime.strptime('8/18/2008', date_format)
b = datetime.strptime('9/26/2008', date_format)
delta = b - a
print delta.days # that's it

на самом деле, в этом случае будет более подходящим класс date, чем datetime.

Jeremy Cantrell 30.09.2008 19:08

@JeremyCantrell И все же, даже восемь лет спустя, date все еще не имеет собственного эквивалента strptime().

JAB 12.02.2016 17:57

Зачем нужен strptime аргумент format? Должна быть ясна дата первого аргумента, имеющая формат.

Timo 19.05.2018 14:59

Дни до Рождества:

>>> import datetime
>>> today = datetime.date.today()
>>> someday = datetime.date(2008, 12, 25)
>>> diff = someday - today
>>> diff.days
86

Больше арифметики здесь.

Вам нужен модуль datetime.

>>> from datetime import datetime, timedelta 
>>> datetime(2008,08,18) - datetime(2008,09,26) 
datetime.timedelta(4) 

Другой пример:

>>> import datetime 
>>> today = datetime.date.today() 
>>> print(today)
2008-09-01 
>>> last_year = datetime.date(2007, 9, 1) 
>>> print(today - last_year)
366 days, 0:00:00 

Как указал здесь

Как я могу получить это без части 0:00:00?

Vicki B 13.09.2019 02:17

@VickiB delta = today - last_yearprint(delta.days)

dbakiu 07.01.2020 16:18

Обратите внимание на порядок вычислений: from_earlier_time - to_later_time, тогда вы получите положительное значение timedelta! Ни в коем случае. Странно как-то.

Jerry T 01.01.2021 03:25

from datetime import datetime
start_date = datetime.strptime('8/18/2008', "%m/%d/%Y")
end_date = datetime.strptime('9/26/2008', "%m/%d/%Y")
print abs((end_date-start_date).days)

Это не добавляет ничего нового по сравнению с ответами, данными 4 годами ранее. -1.

Mark Amery 27.06.2017 02:16

+1 за использование abs(), что полезно, когда сравниваемые даты заранее неизвестны и это разница, которая вас интересует. Если ваша вторая дата в datetime.strptime(date, date) позже первой даты, результат будет отрицательным. abs() делает все входные данные абсолютными (т. Е. Положительными).

veuncent 05.07.2018 23:06

from datetime import date
def d(s):
  [month, day, year] = map(int, s.split('/'))
  return date(year, month, day)
def days(start, end):
  return (d(end) - d(start)).days
print days('8/18/2008', '9/26/2008')

Это, конечно, предполагает, что вы уже подтвердили, что ваши даты находятся в формате r'\d+/\d+/\d+'.

Это не добавляет ничего нового по сравнению с ответами, данными 8 годами ранее. -1.

Mark Amery 27.06.2017 02:17

Основное различие заключается в том, что большинство других ответов даже не удосужились учесть тот факт, что OP имел свои даты в виде строк. И те, кто это делал, в основном использовали более сложные форматеры, чем это было необходимо. Итак, основное отличие - map(int, s.split('/')). Не совсем новаторский, но опять же, этот вопрос довольно глупый, простой. Мой ответ просто показывает другой способ снять шкуру с кошки.

Parthian Shot 28.06.2017 03:05

Также упоминается проверка того, что даты находятся в правильном формате, и дается регулярное выражение проверки в первом приближении. Что другие не сделали.

Parthian Shot 28.06.2017 03:07

Это также легко сделать с помощью arrow:

import arrow

a = arrow.get('2017-05-09')
b = arrow.get('2017-05-11')

delta = (b-a)
print delta.days

Для справки: http://arrow.readthedocs.io/en/latest/

Для расчета даты и времени есть несколько вариантов, но я напишу простой способ:

from datetime import timedelta, datetime, date
import dateutil.relativedelta

# current time
date_and_time = datetime.now()
date_only = date.today()
time_only = datetime.now().time()

# calculate date and time
result = date_and_time - timedelta(hours=26, minutes=25, seconds=10)

# calculate dates: years (-/+)
result = date_only - dateutil.relativedelta.relativedelta(years=10)

# months
result = date_only - dateutil.relativedelta.relativedelta(months=10)

# days
result = date_only - dateutil.relativedelta.relativedelta(days=10)

# calculate time 
result = date_and_time - timedelta(hours=26, minutes=25, seconds=10)
result.time()

Надеюсь, это поможет

без использования Lib только чистый код:

#Calculate the Days between Two Date

daysOfMonths = [ 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]

def isLeapYear(year):

    # Pseudo code for this algorithm is found at
    # http://en.wikipedia.org/wiki/Leap_year#Algorithm
    ## if (year is not divisible by 4) then (it is a common Year)
    #else if (year is not divisable by 100) then (ut us a leap year)
    #else if (year is not disible by 400) then (it is a common year)
    #else(it is aleap year)
    return (year % 4 == 0 and year % 100 != 0) or year % 400 == 0

def Count_Days(year1, month1, day1):
    if month1 ==2:
        if isLeapYear(year1):
            if day1 < daysOfMonths[month1-1]+1:
                return year1, month1, day1+1
            else:
                if month1 ==12:
                    return year1+1,1,1
                else:
                    return year1, month1 +1 , 1
        else: 
            if day1 < daysOfMonths[month1-1]:
                return year1, month1, day1+1
            else:
                if month1 ==12:
                    return year1+1,1,1
                else:
                    return year1, month1 +1 , 1
    else:
        if day1 < daysOfMonths[month1-1]:
             return year1, month1, day1+1
        else:
            if month1 ==12:
                return year1+1,1,1
            else:
                    return year1, month1 +1 , 1


def daysBetweenDates(y1, m1, d1, y2, m2, d2,end_day):

    if y1 > y2:
        m1,m2 = m2,m1
        y1,y2 = y2,y1
        d1,d2 = d2,d1
    days=0
    while(not(m1==m2 and y1==y2 and d1==d2)):
        y1,m1,d1 = Count_Days(y1,m1,d1)
        days+=1
    if end_day:
        days+=1
    return days


# Test Case

def test():
    test_cases = [((2012,1,1,2012,2,28,False), 58), 
                  ((2012,1,1,2012,3,1,False), 60),
                  ((2011,6,30,2012,6,30,False), 366),
                  ((2011,1,1,2012,8,8,False), 585 ),
                  ((1994,5,15,2019,8,31,False), 9239),
                  ((1999,3,24,2018,2,4,False), 6892),
                  ((1999,6,24,2018,8,4,False),6981),
                  ((1995,5,24,2018,12,15,False),8606),
                  ((1994,8,24,2019,12,15,True),9245),
                  ((2019,12,15,1994,8,24,True),9245),
                  ((2019,5,15,1994,10,24,True),8970),
                  ((1994,11,24,2019,8,15,True),9031)]

    for (args, answer) in test_cases:
        result = daysBetweenDates(*args)
        if result != answer:
            print "Test with data:", args, "failed"
        else:
            print "Test case passed!"

test()

Вот три способа решить эту проблему:

from datetime import datetime

Now = datetime.now()
StartDate = datetime.strptime(str(Now.year) +'-01-01', '%Y-%m-%d')
NumberOfDays = (Now - StartDate)

print(NumberOfDays.days)                     # Starts at 0
print(datetime.now().timetuple().tm_yday)    # Starts at 1
print(Now.strftime('%j'))                    # Starts at 1

Также существует метод datetime.toordinal(), о котором еще не упоминалось:

import datetime
print(datetime.date(2008,9,26).toordinal() - datetime.date(2008,8,18).toordinal())  # 39

https://docs.python.org/3/library/datetime.html#datetime.date.toordinal

date.toordinal()

Return the proleptic Gregorian ordinal of the date, where January 1 of year 1 has ordinal 1. For any date object d, date.fromordinal(d.toordinal()) == d.

Кажется, хорошо подходит для расчета разницы в днях, хотя и не так удобочитаем, как timedelta.days.

Бывают случаи, когда этот подход выигрывает. Например, фактическая разница между 23.07.2019 23:50 и 00:10 10.07.2019 составляет двадцать минут. (d1 - d0).days возвращает 0, d1.toordinal() - d0.toordinal() возвращает 1. Зависит от того, что вам нужно в вашем фактическом сценарии использования.

peter.slizik 10.07.2019 11:52

этот подход может фактически сравнивать datetime и date. Например, чтобы проверить, если 2020-04-17 == 2020-04017 00:00:00

Harry Duong 17.04.2020 04:04

все ответили отлично, используя дату, позвольте мне попытаться ответить на него с помощью панд

dt = pd.to_datetime('2008/08/18', format='%Y/%m/%d')
dt1 = pd.to_datetime('2008/09/26', format='%Y/%m/%d')

(dt1-dt).days

Это даст ответ. В случае, если один из входных данных - столбец фрейма данных. просто используйте dt.days вместо дней

(dt1-dt).dt.days

Без использования объекта datetime в Python.

# A date has day 'd', month 'm' and year 'y' 
class Date:
    def __init__(self, d, m, y):
            self.d = d
            self.m = m
            self.y = y

# To store number of days in all months from 
# January to Dec. 
monthDays = [31, 28, 31, 30, 31, 30,
                                            31, 31, 30, 31, 30, 31 ]

# This function counts number of leap years 
# before the given date 
def countLeapYears(d):

    years = d.y

    # Check if the current year needs to be considered 
    # for the count of leap years or not 
    if (d.m <= 2) :
            years-= 1

    # An year is a leap year if it is a multiple of 4, 
    # multiple of 400 and not a multiple of 100. 
    return int(years / 4 - years / 100 + years / 400 )


# This function returns number of days between two 
# given dates 
def getDifference(dt1, dt2) :

    # COUNT TOTAL NUMBER OF DAYS BEFORE FIRST DATE 'dt1' 

    # initialize count using years and day 
    n1 = dt1.y * 365 + dt1.d

    # Add days for months in given date 
    for i in range(0, dt1.m - 1) :
            n1 += monthDays[i]

    # Since every leap year is of 366 days, 
    # Add a day for every leap year 
    n1 += countLeapYears(dt1)

    # SIMILARLY, COUNT TOTAL NUMBER OF DAYS BEFORE 'dt2' 

    n2 = dt2.y * 365 + dt2.d
    for i in range(0, dt2.m - 1) :
            n2 += monthDays[i]
    n2 += countLeapYears(dt2)

    # return difference between two counts 
    return (n2 - n1)


# Driver program 
dt1 = Date(31, 12, 2018 )
dt2 = Date(1, 1, 2019 )

print(getDifference(dt1, dt2), "days")

-1: непроверенный код. dt = Date(01, 01, 2019 ) SyntaxError: leading zeros in decimal integer literals are not permitted; use an 0o prefix for octal integers. Даже если я исправил эту ошибку, было выброшено IndexError.

Jay Lee 24.01.2021 16:58

Другие вопросы по теме