Python 如何计算当前时间减少或增加一个月

本文探讨了在Python中处理日期时遇到的一个常见错误:尝试将一个月减少到不存在的日期。通过分析错误原因,介绍了如何使用dateutil库的relativedelta函数正确地实现日期的减月操作。

 

问题

今天在之前的代码中发现了一个bug,有个计算当前时间减少一个月的函数,其报出下面的异常信息:

ValueError: day is out of range for month

看一下代码:

import datatime
def _last_month(now_time):
    last_month = now_time.month - 1
    last_year = now_time.year
    if last_month == 0:
        last_month = 12
        last_year -= 1
    month_time = datetime.datetime(month=last_month, year=last_year, day=now_time.day)
    return month_time

原因

问题出现在day=now_time.day上。后来想了一下,发现问题原因是3月30日减少一个月是2月30日,而2月没有30日,所以就抛出了上面的异常信息。


解决办法

对于日期操作,网上的写法都不太一样,而且不确定存在什么bug。日期函数是靠时间来验证的,没准一年以后就出现了(我这个bug是在指定的3月29日以后才能出现,神奇不:D)。
所以我找了一个现有的日期扩展库,希望别人已经踩过大部分坑了。代码如下

import datetime
from dateutil.relativedelta import relativedelta

if __name__ == "__main__":
    print(datetime.date.today() - relativedelta(months=+1))

可以看出,主要是使用relativedelta类。初始化参数months是月的差异。
安装这个库也很简单,执行命令pip install python-dateutil


源码分析

代码在
https://github.com/dateutil/dateutil/blob/master/dateutil/relativedelta.py

判断应该是在354行开始:

if self.months:
            assert 1 <= abs(self.months) <= 12
            month += self.months
            if month > 12:
                year += 1
                month -= 12
            elif month < 1:
                year -= 1
                month += 12
        day = min(calendar.monthrange(year, month)[1],
                  self.day or other.day)

 

参考:
https://dateutil.readthedocs.io/en/stable/
https://github.com/dateutil/dateutil

 

### Python中的时间计算Python中,可以利用`datetime`模块来进行各种时间操作和计算。该模块提供了处理日期和时间的功能,并允许开发者执行诸如获取当前时间计算两个时间之间的差值以及格式化时间等常见任务。 以下是关于如何使用`datetime`模块的一些基本概念和示例: #### 获取当前时间和日期 通过调用`datetime.now()`函数可以获得当前时间和日期对象。 ```python from datetime import datetime current_datetime = datetime.now() print(f"Current date and time is {current_datetime}") ``` 此代码片段展示了如何打印出程序运行时的当前日期和时间[^4]。 #### 计算两个日期之间的时间差 如果要计算两个特定日期间相差了多少天者秒数,则可以通过创建两个`datetime`对象并相减得到一个`timedelta`对象。 ```python from datetime import datetime date_format = "%Y-%m-%d %H:%M:%S" a = datetime.strptime('2023-01-01 12:00:00', date_format) b = datetime.strptime('2023-01-02 18:30:00', date_format) delta = b - a print(f"Difference is {delta.days} days, {delta.seconds} seconds") ``` 上述例子说明了怎样解析字符串形式的日期,并求得它们间的差异[^5]。 #### 设置固定偏移量增加减少时间 还可以借助`timedelta`类轻松实现向现有时间上加减一定数量的日历单位(如日、小时等)。 ```python from datetime import timedelta, datetime future_date = datetime.now() + timedelta(days=7) past_date = datetime.now() - timedelta(hours=48) print(f"A week from now will be {future_date}. Two days ago was {past_date}.") ``` 这里演示的是如何基于今天的日期向前推七天后的日期是回溯两天前的具体时刻[^6]。 #### 格式化输出时间 最后一点值得注意的就是能够按照自定义模式转换成易于阅读的形式展示给最终用户查看。 ```python formatted_time = current_datetime.strftime("%A, %B %d, %Y %I:%M%p") print(formatted_time) ``` 这段脚本解释了把标准内部表示转化为更友好外观的过程[^7]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值