จะคำนวณจำนวนวันระหว่างวันที่สองวันได้อย่างไร?


506

หากฉันมีสองวัน (เช่น'8/18/2008'และ'9/26/2008') วิธีที่ดีที่สุดในการรับจำนวนวันระหว่างสองวันนี้คืออะไร

คำตอบ:


800

หากคุณมีวัตถุสองวันคุณสามารถลบ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

ดูคำตอบนี้สำหรับตัวอย่างอื่น


2
คำตอบที่ดีที่นี่ ตั้งแต่จำนวนมากของคนที่อาจจะมีการใช้หมีแพนด้ากรอบข้อมูลที่คิดว่าอาจจะมีประโยชน์ในการตรวจสอบการเชื่อมโยงกับวิธีการแปลงจากnp.datetime64การpython datetime stackoverflow.com/questions/52982056/...
Pramit

สิ่งที่ดีคือโซลูชันนี้จะส่งคืนเดลต้าที่ถูกต้องสำหรับปีอธิกสุรทิน
Lasma

154

การใช้พลังของ 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

4
อันที่จริงคลาสวันที่จะเหมาะสมกว่าในกรณีนี้มากกว่า datetime
Jeremy Cantrell

10
@JeremyCantrell และยังแม้แปดปีต่อมายังขาดเทียบเท่าของตัวเองเพื่อdate strptime()
JAB

ทำไมต้องstrptimeมีformatarg ควรมีความชัดเจนด้วยวันที่หาเรื่องแรกที่มีรูปแบบ
Timo

36

วันก่อนวันคริสต์มาส:

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

คณิตศาสตร์เพิ่มเติมที่นี่


16

คุณต้องการโมดูล 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 

ดังที่ได้กล่าวไว้ที่นี่


1
ฉันจะได้รับสิ่งนี้ได้อย่างไรหากไม่มีส่วน 0:00:00
Vicki B

@VickiBdelta = today - last_year print(delta.days)
dbakiu

8
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)

2
สิ่งนี้ไม่ได้เพิ่มอะไรใหม่เมื่อเทียบกับคำตอบที่ให้ไว้ 4 ปี -1
Mark Amery

+1 สำหรับการใช้งานabs()ซึ่งมีประโยชน์เมื่อไม่ทราบวันที่เปรียบเทียบไว้ล่วงหน้าและเป็นความแตกต่างที่คุณสนใจหากวันที่ที่สองของคุณdatetime.strptime(date, date)ช้ากว่าวันที่แรกผลลัพธ์จะเป็นค่าลบ abs()ทำให้การป้อนข้อมูลทั้งหมดแน่นอน (เช่น. บวก)
veuncent


6

โดยไม่ต้องใช้ 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()

3

ทุกคนตอบกลับอย่างดีเยี่ยมโดยใช้วันที่ให้ฉันลองตอบโดยใช้แพนด้า

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

นี้จะให้คำตอบ ในกรณีที่หนึ่งในอินพุตคือคอลัมน์ dataframe ใช้dt.daysแทนวัน

(dt1-dt).dt.days

2

นอกจากนี้ยังมี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 ()

กลับลำดับเกรโกเรียนวันที่ 1 มกราคมของปีที่ 1 มีลำดับที่ 1. สำหรับใด ๆdateวัตถุddate.fromordinal(d.toordinal()) == d ,

ดูเหมือนว่าเหมาะสำหรับการคำนวณวันที่แตกต่าง timedelta.daysแต่ไม่เป็นอ่านเป็น


1
มีหลายกรณีที่วิธีการนี้ชนะ ตัวอย่างเช่นความแตกต่างที่แท้จริงระหว่าง 2019-07-09 23:50 และ 2019-07-10 00:10 คือยี่สิบนาที (d1 - d0).daysผลตอบแทน0, ผลตอบแทนd1.toordinal() - d0.toordinal() 1ขึ้นอยู่กับสิ่งที่คุณต้องการใน usecase ที่แท้จริงของคุณ
peter.slizik

วิธีนี้สามารถเปรียบเทียบวันที่และเวลาและวันที่ได้ ตัวอย่างเช่นเพื่อตรวจสอบว่า 2020-04-17 == 2020-04017 00:00:00
Harry Duong

2

สำหรับการคำนวณวันที่และเวลามีหลายตัวเลือก แต่ฉันจะเขียนวิธีง่าย ๆ :

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

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

# calculate date and time
result = date_and_time - datetime.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 - datetime.timedelta(hours=26, minutes=25, seconds=10)
result.time()

หวังว่ามันจะช่วย


1

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+'นี้จะถือว่าแน่นอนว่าคุณได้ยืนยันแล้วว่าวันที่ที่คุณอยู่ในรูปแบบ


1
สิ่งนี้ไม่ได้เพิ่มอะไรใหม่เมื่อเทียบกับคำตอบที่ให้เมื่อ 8 ปีก่อน -1
Mark Amery

1
ความแตกต่างที่สำคัญคือคำตอบอื่น ๆ ส่วนใหญ่ไม่ได้สนใจที่จะคำนึงถึงความจริงที่ว่า OP มีวันที่ของเขาเป็นสตริง และคนที่คิดว่าส่วนใหญ่ใช้ฟอร์แมตเตอร์ที่ซับซ้อนกว่าที่จำเป็นอย่างเคร่งครัด map(int, s.split('/'))ดังนั้นความแตกต่างที่สำคัญคือ ไม่ได้แหวกแนว แต่จากนั้นคำถามนี้อีกครั้งเป็นขั้นพื้นฐานที่ค่อนข้างโง่ คำตอบของฉันเพียงแสดงอีกวิธีหนึ่งในการดูแลแมว
คู่ปรับ Shot

นอกจากนี้ยังกล่าวถึงการตรวจสอบว่าวันที่อยู่ในรูปแบบที่ถูกต้องและให้ regex การตรวจสอบแรกประมาณ ซึ่งคนอื่นไม่ได้
คู่ปรับ Shot

1

ต่อไปนี้เป็นสามวิธีในการแก้ไขปัญหานี้:

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
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.