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

问题

今天在之前的代码中发现了一个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

@完

posted @ 2017-03-29 18:26  kris_zhang  阅读(33397)  评论(0编辑  收藏  举报