如何从日期中减去一天?

1163

我有一个Python的datetime.datetime对象,最好的方法是什么来减去一天?


1
查找Python中两个日期之间是否相隔24小时的问题。 - jfs
这个线程直接用于将一年中的某一天转换为日期。 - Hari
8个回答

1950
您可以使用一个 timedelta 对象:
from datetime import datetime, timedelta
    
d = datetime.today() - timedelta(days=days_to_subtract)

35
如果你不忽略时区,那么答案就更加复杂。 - jfs
1
另外,你如何将其与特定日期相关联?请参阅我的问题:http://stackoverflow.com/questions/43092508/first-available-date-previous-day-month-year - WJA
1
它也可以与其他单位一起使用,例如我曾经使用过 timedelta(minutes=12) - Nagev
文档说明将返回“表示两个日期、时间或日期时间实例之间差异的持续时间...”,那么如何获取5天前或5天后的实际日期? - oldboy
1
@jfs 我认为这个答案在时区方面是可以的。如果你减去一天,那么可能会减去23或25个小时,但时间组件将保持不变。这就是我所期望的正常行为。夏令时(DST)使得一天不总是24小时。 - at54321
@at54321:是否需要确切的24小时,或者拥有完全相同的时钟时间(小时)更可取,可能取决于任务。[我的答案涵盖了两种情况](https://dev59.com/43RC5IYBdhLWcg3wAcM3#25427822) - jfs

106

减去 datetime.timedelta(days=1)


95

如果你的 Python datetime 对象是有时区信息的,那么你需要小心避免在 DST 转换(或其他原因导致 UTC 偏移量变化)期间出现错误:

from datetime import datetime, timedelta
from tzlocal import get_localzone # pip install tzlocal

DAY = timedelta(1)
local_tz = get_localzone()   # get local timezone
now = datetime.now(local_tz) # get timezone-aware datetime object
day_ago = local_tz.normalize(now - DAY) # exactly 24 hours ago, time may differ
naive = now.replace(tzinfo=None) - DAY # same time
yesterday = local_tz.localize(naive, is_dst=None) # but elapsed hours may differ

通常情况下,如果本地时区的UTC偏移在过去的一天内发生了改变,那么day_agoyesterday可能会不同。

例如,美国/洛杉矶时区的夏令时/夏季时间将于2014年11月2日星期日凌晨02:00:00结束,因此如果:

import pytz # pip install pytz

local_tz = pytz.timezone('America/Los_Angeles')
now = local_tz.localize(datetime(2014, 11, 2, 10), is_dst=None)
# 2014-11-02 10:00:00 PST-0800

那么day_agoyesterday是有区别的:

  • day_ago 恰好是 24 小时前(相对于 now),但是是在上午 11 点,而不是像 now 的上午 10 点。
  • yesterday 是昨天上午 10 点,但相对于 now 是 25 小时前,而不是 24 小时。

pendulum 模块 可以自动处理这个问题:

>>> import pendulum  # $ pip install pendulum

>>> now = pendulum.create(2014, 11, 2, 10, tz='America/Los_Angeles')
>>> day_ago = now.subtract(hours=24)  # exactly 24 hours ago
>>> yesterday = now.subtract(days=1)  # yesterday at 10 am but it is 25 hours ago

>>> (now - day_ago).in_hours()
24
>>> (now - yesterday).in_hours()
25

>>> now
<Pendulum [2014-11-02T10:00:00-08:00]>
>>> day_ago
<Pendulum [2014-11-01T11:00:00-07:00]>
>>> yesterday
<Pendulum [2014-11-01T10:00:00-07:00]>

53

只是为了详细说明一种替代方法和它有用的用例:

  • 从当前日期时间减去1天:
from datetime import datetime, timedelta
print datetime.now() + timedelta(days=-1)  # Here, I am adding a negative timedelta
  • 有用的情况: 如果您想从当前日期时间中添加5天并减去5小时。即,现在时间往后推5天但少5小时后是什么日期时间?
from datetime import datetime, timedelta
print datetime.now() + timedelta(days=5, hours=-5)
它同样可以与其他参数一起使用,例如秒、周等。

14

还有另一个我喜欢使用的好函数,用于计算上个月的第一天/最后一天或其他相对的时间差等等...

relativedelta函数来自于dateutil库(datetime库的强大扩展)。

import datetime as dt
from dateutil.relativedelta import relativedelta
#get first and last day of this and last month)
today = dt.date.today()
first_day_this_month = dt.date(day=1, month=today.month, year=today.year)
last_day_last_month = first_day_this_month - relativedelta(days=1)
print (first_day_this_month, last_day_last_month)

>2015-03-01 2015-02-28

8
优秀的箭头模块已经存在。
import arrow
utc = arrow.utcnow()
utc_yesterday = utc.shift(days=-1)
print(utc, '\n', utc_yesterday)

输出:

2017-04-06T11:17:34.431397+00:00 
 2017-04-05T11:17:34.431397+00:00

1

你还可以使用pandas的pd.Timedelta('1D')(它非常灵活,甚至可以传入像pd.Timedelta('1d 5h 9s')这样的参数,表示1天、5小时和9秒)。

pandas的一个方便之处在于它的日期时间对象是基于datetime.datetime构建的,因此任何涉及Python的datetime对象的操作都可以在pandas的日期时间对象上正常运行,反之亦然。

import pandas as pd
import numpy as np
from datetime import datetime, date, timedelta

datetime.now() - pd.Timedelta('1d')           # datetime.datetime(2023, 2, 21, 15, 35, 23, 603832)
pd.Timestamp('now') - timedelta(days=1)       # Timestamp('2023-02-21 15:35:23.741866')
pd.Timestamp('now') - pd.Timedelta('1d')      # Timestamp('2023-02-21 15:35:23.882746')
pd.Timestamp('now') - np.timedelta64(1, 'D')  # Timestamp('2023-02-21 15:35:24.032356')

date(2022, 2, 22) - pd.Timedelta('10d')        # datetime.date(2022, 2, 12)

pandas的优势在于您可以执行矢量化操作(即使dtype为object)。您可以使用pd.Timedelta/datetime.timedelta/np.timedelta64中的任何一个。

pd.Series([datetime(2023,2,22), datetime(2023,2,21), datetime(2023,2,20)]) - pd.Timedelta('1d')
pd.Series([date(2023,2,22), date(2023,2,21), date(2023,2,20)]) - timedelta(days=1)
pd.Series([date(2023,2,22), date(2023,2,21), date(2023,2,20)]) - np.timedelta64(1, 'D')

-15
class myDate:

    def __init__(self):
        self.day = 0
        self.month = 0
        self.year = 0
        ## for checking valid days month and year
        while (True):
            d = int(input("Enter The day :- "))
            if (d > 31):
                print("Plz 1 To 30 value Enter ........")
            else:
                self.day = d
                break

        while (True):
            m = int(input("Enter The Month :- "))
            if (m > 13):
                print("Plz 1 To 12 value Enter ........")
            else:
                self.month = m
                break

        while (True):
            y = int(input("Enter The Year :- "))
            if (y > 9999 and y < 0000):
                print("Plz 0000 To 9999 value Enter ........")
            else:
                self.year = y
                break
    ## method for aday ands cnttract days
    def adayDays(self, n):
        ## aday days to date day
        nd = self.day + n
        print(nd)
        ## check days subtract from date
        if nd == 0: ## check if days are 7  subtracted from 7 then,........
            if(self.year % 4 == 0):
                if(self.month == 3):
                    self.day = 29
                    self.month -= 1
                    self.year = self. year
            else:
                if(self.month == 3):
                    self.day = 28
                    self.month -= 1
                    self.year = self. year
            if  (self.month == 5) or (self.month == 7) or (self.month == 8) or (self.month == 10) or (self.month == 12):
                self.day = 30
                self.month -= 1
                self.year = self. year
                   
            elif (self.month == 2) or (self.month == 4) or (self.month == 6) or (self.month == 9) or (self.month == 11):
                self.day = 31
                self.month -= 1
                self.year = self. year

            elif(self.month == 1):
                self.month = 12
                self.year -= 1    
        ## nd == 0 if condition over
        ## after subtract days to day io goes into negative then
        elif nd < 0 :   
            n = abs(n)## return positive if no is negative
            for i in range (n,0,-1): ## 
                
                if self.day == 0:

                    if self.month == 1:
                        self.day = 30
                        
                        self.month = 12
                        self.year -= 1
                    else:
                        self.month -= 1
                        if(self.month == 1) or (self.month == 3)or (self.month == 5) or (self.month == 7) or (self.month == 8) or (self.month == 10) or (self.month ==12):
                            self.day = 30
                        elif(self.month == 4)or (self.month == 6) or (self.month == 9) or (self.month == 11):
                            self.day = 29
                        elif(self.month == 2):
                            if(self.year % 4 == 0):
                                self.day == 28
                            else:
                                self.day == 27
                else:
                    self.day -= 1

        ## enf of elif negative days
        ## adaying days to DATE
        else:
            cnt = 0
            while (True):

                if self.month == 2:  # check leap year
                    
                    if(self.year % 4 == 0):
                        if(nd > 29):
                            cnt = nd - 29
                            nd = cnt
                            self.month += 1
                        else:
                            self.day = nd
                            break
                ## if not leap year then
                    else:  
                    
                        if(nd > 28):
                            cnt = nd - 28
                            nd = cnt
                            self.month += 1
                        else:
                            self.day = nd
                            break
                ## checking month other than february month
                elif(self.month == 1) or (self.month == 3) or (self.month == 5) or (self.month == 7) or (self.month == 8) or (self.month == 10) or (self.month == 12):
                    if(nd > 31):
                        cnt = nd - 31
                        nd = cnt

                        if(self.month == 12):
                            self.month = 1
                            self.year += 1
                        else:
                            self.month += 1
                    else:
                        self.day = nd
                        break

                elif(self.month == 4) or (self.month == 6) or (self.month == 9) or (self.month == 11):
                    if(nd > 30):
                        cnt = nd - 30
                        nd = cnt
                        self.month += 1

                    else:
                        self.day = nd
                        break
                ## end of month condition
        ## end of while loop
    ## end of else condition for adaying days
    def formatDate(self,frmt):

        if(frmt == 1):
            ff=str(self.day)+"-"+str(self.month)+"-"+str(self.year)
        elif(frmt == 2):
            ff=str(self.month)+"-"+str(self.day)+"-"+str(self.year)
        elif(frmt == 3):
            ff =str(self.year),"-",str(self.month),"-",str(self.day)
        elif(frmt == 0):
            print("Thanky You.....................")
            
        else:
            print("Enter Correct Choice.......")
        print(ff)
            
            

dt = myDate()
nday = int(input("Enter No. For Aday or SUBTRACT Days :: "))
dt.adayDays(nday)
print("1 : day-month-year")
print("2 : month-day-year")
print("3 : year-month-day")
print("0 : EXIT")
frmt = int (input("Enter Your Choice :: "))
dt.formatDate(frmt)

    enter code here

2
尽可能使用标准库更好,因为它通常由许多人维护,错误较少且执行速度更快。 - Gabriel Avendaño
日期时间库在这里发挥作用 :) 不需要特殊逻辑 - Ido Bleicher

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接