如何在 Python 中获取下个月的第一天?

How can I get the first day of the next month in Python?

如何获取Python下个月的第一天?例如,如果现在是 2019-12-31,则下个月的第一天是 2020-01-01。如果现在是2019-08-01,那么下个月的第一天就是2019-09-01。

我想到了这个:

import datetime

def first_day_of_next_month(dt):
    '''Get the first day of the next month. Preserves the timezone.

    Args:
        dt (datetime.datetime): The current datetime

    Returns:
        datetime.datetime: The first day of the next month at 00:00:00.
    '''
    if dt.month == 12:
        return datetime.datetime(year=dt.year+1,
                                 month=1,
                                 day=1,
                                 tzinfo=dt.tzinfo)
    else:
        return datetime.datetime(year=dt.year,
                                 month=dt.month+1,
                                 day=1,
                                 tzinfo=dt.tzinfo)


# Example usage (assuming that today is 2021-01-28):
first_day_of_next_month(datetime.datetime.now())
# Returns: datetime.datetime(2021, 2, 1, 0, 0)

是否正确?有没有更好的方法?

您可以使用 calendar 获取给定月份的天数,然后添加 timedelta(days=...),如下所示:

from datetime import date, timedelta
from calendar import monthrange

days_in_month = lambda dt: monthrange(dt.year, dt.month)[1]
today = date.today()
first_day = today.replace(day=1) + timedelta(days_in_month(today))
print(first_day)

如果您不介意外部部门,您可以使用 dateutil(我喜欢...)

from datetime import date
from dateutil.relativedelta import relativedelta

today = date.today()
first_day = today.replace(day=1) + relativedelta(months=1)
print(first_day)

你的方法看起来不错,但我会这样做:

import datetime
from dateutil import relativedelta

dt = datetime.datetime(year=1998,
                             month=12,
                             day=12)

nextmonth = dt + relativedelta.relativedelta(months=1)
nextmonth.replace(day=1)
print(nextmonth)

使用dateutil你可以做到最字面的可能:

import datetime
from dateutil import relativedelta
today = datetime.date.today()

next_month = today + relativedelta.relativedelta(months=1, day=1)

英文:在今天的日期上加上1个月,并将(该月的)日期设置为1。注意day(s)month(s)的单复数形式的用法.单数设置属性为一个值,复数加上句点数。

您可以将此 relativedelta.relativedelta 对象存储到一个变量中并传递它。其他答案涉及更多的编程逻辑。

编辑你也可以用标准的datetime库来做,但它不是那么漂亮:

next_month = (today.replace(day=1) + datetime.timedelta(days=32)).replace(day=1)

将日期设置为当月的 1 号,加上 32 天(或 31 到 59 之间的任何数字,保证跳到下个月),然后将日期设置为该月的 1 号。

这是仅使用标准 datetime 库的单行解决方案:

(dt.replace(day=1) + datetime.timedelta(days=32)).replace(day=1)

示例:

>>> dt = datetime.datetime(2016, 2, 29)
>>> print((dt.replace(day=1) + datetime.timedelta(days=32)).replace(day=1))
2016-03-01 00:00:00

>>> dt = datetime.datetime(2019, 12, 31)
>>> print((dt.replace(day=1) + datetime.timedelta(days=32)).replace(day=1))
2020-01-01 00:00:00

>>> dt = datetime.datetime(2019, 12, 1)
>>> print((dt.replace(day=1) + datetime.timedelta(days=32)).replace(day=1))
2020-01-01 00:00:00

仅使用 python 个标准库:

import datetime
today = datetime.date.today()
first_of_next_month = return date.replace(
    day=1,
    month=date.month % 12 + 1,
    year=date.year + (date.month // 12)
)

可以概括为...

def get_first_of_month(date, month_offset=0):
    # zero based indexing of month to make math work
    month_count = date.month - 1 + month_offset
    return date.replace(
        day=1, month=month_count % 12 + 1, year=date.year + (month_count // 12)
    )

first_of_next_month = get_first_of_month(today, 1)

不需要第 3 方库的其他解决方案包括:

  • 是另一个不错的选择。
  • 如果准确的timedelta对您有帮助, 对 稍作修改可能会很方便:
    import calendar, datetime
    
    today = datetime.date.today()
    time_until_next_month = datetime.timedelta(
        calendar.monthrange(today.year, today.month)[1] - today.day + 1
    )
    first_of_next_month = today + time_until_next_month
    

使用 Zope 的 DateTime 库可以实现非常简单的解决方案

from DateTime.DateTime import DateTime

date = DateTime()  # today

while date.day() != 1:
    date += 1

print(date)

我看到这个问题有很多很好的解决方案,当我偶然发现这个问题时,我个人正在寻找一个解决方案来获取上个月的第一天和最后一天。

但这里有一个我认为非常简单和优雅的解决方案:

date = datetime.datetime.now().date()
same_time_next_month = date + datetime.timedelta(days = date.day)
first_day_of_next_month_from_date = same_time_next_month - datetime.timedelta(days = same_time_next_month.day - 1)

这里我们简单的把目标日期的天加上日期得到下个月的同一时间,然后去掉得到的新日期经过的天数。

提取年月,加1,用年月日=1组成新的日期:

from datetime import date

now       = date(2020,12,18)
y,m       = divmod(now.year*12+now.month,12)
nextMonth = date(y,m+1,1)

print(now,nextMonth)
# 2020-12-18 2021-01-01

试试这个,对于每个月的开始日,将 MonthEnd(1) 更改为 MonthBegin(1):

import pandas as pd
from pandas.tseries.offsets import MonthBegin, MonthEnd

date_list = (pd.date_range('2021-01-01', '2022-01-31', 
              freq='MS') + MonthEnd(1)).strftime('%Y-%m-%d').tolist()
date_list

输出:

['2021-01-31',
 '2021-02-28',
 '2021-03-31',
 '2021-04-30',
 '2021-05-31',
 '2021-06-30',
 '2021-07-31',
 '2021-08-31',
 '2021-09-30',
 '2021-10-31',
 '2021-11-30',
 '2021-12-31',
 '2022-01-31']