如何在Python中自定义月份而不使用库增加date时间

我需要增加date时间值的月份

next_month = datetime.datetime(mydate.year, mydate.month+1, 1) 

当月份是12时,变成13,并提出错误“月份必须在1..12”。 (我预计这一年会增加)

我想使用timedelta,但它不需要一个月的参数。 有相对的 Python包,但我不想只为此安装它。 也有使用strtotime的解决scheme。

 time = strtotime(str(mydate)); next_month = date("Ymd", strtotime("+1 month", time)); 

我不想从date时间转换到时间,然后到date时间; 因此,它也是一个图书馆

有没有人像使用timedelta有任何好的和简单的解决scheme?

编辑 – 根据您的date评论需要向下舍入,如果下个月有更less的日子,这里是一个解决scheme:

 >>> import datetime >>> import calendar >>> >>> def add_months(sourcedate,months): ... month = sourcedate.month - 1 + months ... year = int(sourcedate.year + month / 12 ) ... month = month % 12 + 1 ... day = min(sourcedate.day,calendar.monthrange(year,month)[1]) ... return datetime.date(year,month,day) ... >>> somedate = datetime.date.today() >>> somedate datetime.date(2010, 11, 9) >>> add_months(somedate,1) datetime.date(2010, 12, 9) >>> add_months(somedate,23) datetime.date(2012, 10, 9) >>> otherdate = datetime.date(2010,10,31) >>> add_months(otherdate,1) datetime.date(2010, 11, 30) 

另外,如果你不担心小时,分钟和秒钟,你可以使用date而不是datetime 。 如果您担心小时,分钟和秒钟,您需要修改我的代码以使用datetime时间并从源文件复制小时,分钟和秒钟到结果。

这是使用dateutil的relativedelta添加一个月份到一个date的短而甜的方法。

 from datetime import datetime from dateutil.relativedelta import relativedelta date_after_month = datetime.today()+ relativedelta(months=1) print 'Today: ',datetime.today().strftime('%d/%m/%Y') print 'After Month:', date_after_month.strftime('%d/%m/%Y') 

输出:

今天:01/03/2013

在月之后:01/04/2013

警告词 :relativedelta(months = 1)和relativedelta(month = 1)有不同的含义 。

注意:这将需要python-dateutil。 要安装它,你需要在linuxterminal上运行。

 sudo apt-get update && sudo apt-get install python-dateutil 

说明: 在Python中添加月份值

这是我的盐:

 current = datetime.datetime(mydate.year, mydate.month, 1) next_month = datetime.datetime(mydate.year + (mydate.month / 12), ((mydate.month % 12) + 1), 1) 

快速简单:)

因为没有人提出任何解决scheme,这是我迄今为止解决的问题

 year, month= divmod(mydate.month+1, 12) if month == 0: month = 12 year = year -1 next_month = datetime.datetime(mydate.year + year, month, 1) 

使用monthdelta包,它就像timedelta,但日历月,而不是天/小时/等。

这是一个例子:

 from monthdelta import MonthDelta def prev_month(date): """Back one month and preserve day if possible""" return date + MonthDelta(-1) 

比较一下这个DIY方法:

 def prev_month(date): """Back one month and preserve day if possible""" day_of_month = date.day if day_of_month != 1: date = date.replace(day=1) date -= datetime.timedelta(days=1) while True: try: date = date.replace(day=day_of_month) return date except ValueError: day_of_month -= 1 

要计算当前,上个月和下个月:

 import datetime this_month = datetime.date.today().month last_month = datetime.date.today().month - 1 or 12 next_month = (datetime.date.today().month + 1) % 12 or 12 
 from datetime import timedelta try: next = (x.replace(day=1) + timedelta(days=31)).replace(day=x.day) except ValueError: # January 31 will return last day of February. next = (x + timedelta(days=31)).replace(day=1) - timedelta(days=1) 

如果你只是想要下个月的第一天:

 next = (x.replace(day=1) + timedelta(days=31)).replace(day=1) 

也许使用calendar.monthrange()添加当前月份的天数?

 import calendar, datetime def increment_month(when): days = calendar.monthrange(when.year, when.month)[1] return when + datetime.timedelta(days=days) now = datetime.datetime.now() print 'It is now %s' % now print 'In a month, it will be %s' % increment_month(now) 

这个如何? (不需要任何额外的库)

 from datetime import date, timedelta from calendar import monthrange today = date.today() month_later = date(today.year, today.month, monthrange(today.year, today.month)[1]) + timedelta(1) 

对于使用结算的人来说,这个实现可能有一定的价值。

如果您正在使用结算function,那么您可能希望得到“下个月的相同date(如果可能)”,而不是“添加1/12的一年”。

如果你这样做的话,你实际上需要考虑两个值。 否则,在27日以后的任何日子,你都会继续失去几天,直到你在闰年后的第27天结束。

您需要考虑的值:

  • 您想要添加一个月的值
  • 你开始的那天

这样,如果你从第三十一次到第三十次,当你增加一个月时,你会在当天下一个月的时候回到第三十一次。

这是我做到的:

 def closest_date_next_month(year, month, day): month = month + 1 if month == 13: month = 1 year = year + 1 condition = True while condition: try: return datetime.datetime(year, month, day) except ValueError: day = day-1 condition = day > 26 raise Exception('Problem getting date next month') paid_until = closest_date_next_month( last_paid_until.year, last_paid_until.month, original_purchase_date.day) # The trick is here, I'm using the original date, that I started adding from, not the last one 

以及一些调整和使用timedelta在这里我们去:

 from datetime import datetime, timedelta def inc_date(origin_date): day = origin_date.day month = origin_date.month year = origin_date.year if origin_date.month == 12: delta = datetime(year + 1, 1, day) - origin_date else: delta = datetime(year, month + 1, day) - origin_date return origin_date + delta final_date = inc_date(datetime.today()) print final_date.date() 

与Dave Webb的解决scheme类似,但没有那么棘手的模算术:

 import datetime, calendar def increment_month(date): # Go to first of this month, and add 32 days to get to the next month next_month = date.replace(day=1) + datetime.timedelta(32) # Get the day of month that corresponds day = min(date.day, calendar.monthrange(next_month.year, next_month.month)[1]) return next_month.replace(day=day) 

我正在寻找解决相关的问题,find下个月的第一个date, 无论在给定的date的一天。 这1个月后没有find同一天。

所以,如果你想要的只是在2014年12月12日(或12月的任何一天)投放并且在2015年1月1日之前,请试试这个:

 import datetime def get_next_month(date): month = (date.month % 12) + 1 year = date.year + (date.month + 1 > 12) return datetime.datetime(year, month, 1) 

最简单的解决scheme是在月底(我们总是知道月份至less有28天),并且增加足够的时间移动到下一个飞蛾:

 >>> from datetime import datetime, timedelta >>> today = datetime.today() >>> today datetime.datetime(2014, 4, 30, 11, 47, 27, 811253) >>> (today.replace(day=28) + timedelta(days=10)).replace(day=today.day) datetime.datetime(2014, 5, 30, 11, 47, 27, 811253) 

也适用于几年之间:

 >>> dec31 datetime.datetime(2015, 12, 31, 11, 47, 27, 811253) >>> today = dec31 >>> (today.replace(day=28) + timedelta(days=10)).replace(day=today.day) datetime.datetime(2016, 1, 31, 11, 47, 27, 811253) 

请记住,不能保证下个月将会有同一天,例如从1月31日到2月31日,它将会失败:

 >>> today datetime.datetime(2016, 1, 31, 11, 47, 27, 811253) >>> (today.replace(day=28) + timedelta(days=10)).replace(day=today.day) Traceback (most recent call last): File "<stdin>", line 1, in <module> ValueError: day is out of range for month 

所以如果你需要移动到下个月的第一天,这是一个有效的解决scheme,因为你总是知道下个月有第一天( .replace(day=1) )。 否则,要移至最后一天,您可能需要使用:

 >>> today datetime.datetime(2016, 1, 31, 11, 47, 27, 811253) >>> next_month = (today.replace(day=28) + timedelta(days=10)) >>> import calendar >>> next_month.replace(day=min(today.day, calendar.monthrange(next_month.year, next_month.month)[1])) datetime.datetime(2016, 2, 29, 11, 47, 27, 811253) 

只是使用这个:

 import datetime today = datetime.datetime.today() nextMonthDatetime = today + datetime.timedelta(days=(today.max.day - today.day)+1) 

不使用日历的解决scheme:

 def add_month_year(date, years=0, months=0): year, month = date.year + years, date.month + months + 1 dyear, month = divmod(month - 1, 12) rdate = datetime.date(year + dyear, month + 1, 1) - datetime.timedelta(1) return rdate.replace(day = min(rdate.day, date.day)) 
 def add_month(d,n=1): return type(d)(d.year+(d.month+n-1)/12, (d.month+n-1)%12+1, 1) 

这是我想出来的

 from calendar import monthrange def same_day_months_after(start_date, months=1): target_year = start_date.year + ((start_date.month + months) / 12) target_month = (start_date.month + months) % 12 num_days_target_month = monthrange(target_year, target_month)[1] return start_date.replace(year=target_year, month=target_month, day=min(start_date.day, num_days_target_month)) 
 def month_sub(year, month, sub_month): result_month = 0 result_year = 0 if month > (sub_month % 12): result_month = month - (sub_month % 12) result_year = year - (sub_month / 12) else: result_month = 12 - (sub_month % 12) + month result_year = year - (sub_month / 12 + 1) return (result_year, result_month) def month_add(year, month, add_month): return month_sub(year, month, -add_month) >>> month_add(2015, 7, 1) (2015, 8) >>> month_add(2015, 7, 20) (2017, 3) >>> month_add(2015, 7, 12) (2016, 7) >>> month_add(2015, 7, 24) (2017, 7) >>> month_add(2015, 7, -2) (2015, 5) >>> month_add(2015, 7, -12) (2014, 7) >>> month_add(2015, 7, -13) (2014, 6) 

使用时间对象的例子:

 start_time = time.gmtime(time.time()) # start now #increment one month start_time = time.gmtime(time.mktime([start_time.tm_year, start_time.tm_mon+1, start_time.tm_mday, start_time.tm_hour, start_time.tm_min, start_time.tm_sec, 0, 0, 0])) 

我非常简单的解决scheme,它不需要任何额外的模块:

 def addmonth(date): if date.day < 20: date2 = date+timedelta(32) else : date2 = date+timedelta(25) date2.replace(date2.year, date2.month, day) return date2