在 PHP 里,如果想获取下个月的今天,我们通常会用 strtotime('2019-01-13 +1 month') 或者是 strtotime('next month', strtotime('2019-01-13')),结果是 1549987200,用 date 转换成字符串就是 2019-02-13,so easy!

随着时间流逝,到了 2019-01-31 这一天,这代码似乎出问题了,居然得到的是 2019-03-03???

搜了一下,找到鸟哥有一篇文章有解释过为什么会出现这个情况,原因是:

date 内部对于这种事情的处理逻辑:

  1. 先做 +1 month,那么当前是 01-31,加上一以后就是 02-31
  2. 再做日期规范化, 因为 2 月没有 31 号, 所以就好像 2:60 等于 3:00 一样,2019 年 2 月 31 日就等于了 3 月 3 日

知道了原因,那我们就知道该怎么去处理这个问题了。

首先我们先约定,如果是大月的 31 号,遇到小月时则取小月的最后一天作为下个月的今天,为什么这么约定呢,因为在一些结算系统中,假如你第一次月结日是在 31 号,那么下一次月结如果遇到小月,通常是以小月的最后一天作为月结日,而不是跨到再下一个月去月结。

function next_month($date)
{
    if (is_numeric($date)) {
        $date = date('Y-m-d', $date);
    }
    $next_month = date('Y-m-d', strtotime('next month', strtotime($date)));

    // 获取当前日期的年、月、日
    list($d_y, $d_m, $d_d) = explode('-', $date);

    // 获取“一个月后”的年、月、日
    list($n_y, $n_m, $n_d) = explode('-', $next_month);

    // 如果月份差大于 1,说明跨月了,只需要取下一个月的最后一天就可以了;
    // 细心的你可能会想到,如果是 12 月,那么 $n_m - $d_m 就是负数,需不需要再特殊处理呢?
    // 其实是不用的,因为 12 月和 1 月都是大月,不存在“一个月后的今天”是跨到了次年 2 月的情况
    if ($n_m - $d_m > 1) {
        $next_month = date('Y-m-d', strtotime('last day of next month', strtotime($date)));
    }

    return $next_month;
}

这回可以正确地获取到一个月后的日期了,但是,如果我要获取两个月后的今天呢?

echo next_month(next_month('2019-01-31'));

上面这种用法会输出 2019-03-28 这个结果,很明显是不行的,如果是用 date('Y-m-d', strtotime('2019-01-31 +2 months')),在这一次需求里还行,但是如果恰巧遇上两个月后是小月呢?又或者是我要知道第 N 个月后的今天的日期?

其实也很简单,只要对上面的函数稍作修改即可:

function next_months($date, $months = 1) {
        if (is_numeric($date)) {
            $date = date('Y-m-d', $date);
        }
        $next_month = date('Y-m-d', strtotime('+' . $months . ' month', strtotime($date)));
        list($d_y, $d_m, $d_d) = explode('-', $date);
        list($n_y, $n_m, $n_d) = explode('-', $next_month);

        $diff = ($n_y - $d_y) * 12 + ($n_m - $d_m);

        if ($diff > $months) {
            $next_month = date('Y-m-d', strtotime('last day of next month', strtotime($date)));
        }

        return $next_month;
    }

标签: php

已有 2 条评论

  1. JackyCheungs JackyCheungs

    最新的博客居然是一月份的了,你变懒了。

添加新评论