怎么入的坑
我在使用strtotime 获取03月31日对应的上一个月时间的时候,返回的不是2月的时间,而是3月3日,具体原因是为什么?想一探坑的深度,请继续往下看。
strtotime的介绍
在说明问题之前,先简单回顾一下strtotime的基本使用和注意事项,如果对strtotime的使用很了解,可以略过。
实例
<?php
echo(strtotime("now") . "<br>");
echo(strtotime("15 October 1980") . "<br>");
echo(strtotime("+5 hours") . "<br>");
echo(strtotime("+1 week") . "<br>");
echo(strtotime("+1 week 3 days 7 hours 5 seconds") . "<br>");
echo(strtotime("next Monday") . "<br>");
echo(strtotime("last Sunday"));
echo(date('Y-m-d', strtotime("-1 month", strtotime("2021-03-31")));
?>
定义和用法
strtotime() 函数将任何英文文本的日期或时间描述解析为 Unix 时间戳(自 January 1 1970 00:00:00 GMT 起的秒数)。
**注意:**如果年份表示使用两位数格式,则值 0-69 会映射为 2000-2069,值 70-100 会映射为 1970-2000。
**注意:**请注意 m/d/y 或 d-m-y 格式的日期,如果分隔符是斜线(/),则使用美洲的 m/d/y 格式。如果分隔符是横杠(-)或者点(.),则使用欧洲的 d-m-y 格式。为了避免潜在的错误,您应该尽可能使用 YYYY-MM-DD 格式或者使用 date_create_from_format() 函数。
语法
strtotime(time,now);
参数 | 描述 |
---|---|
time | 必需。规定日期/时间字符串。 |
now | 可选。规定用来计算返回值的时间戳。如果省略该参数,则使用当前时间。 |
技术细节
- 返回值:若成功则返回时间戳,失败则返回 FALSE。
- php版本: 4+
找出逻辑规律
- 以下面这个代码为例:
var_dump(date("Y-m-d", strtotime("2017-06-31")));
//输出2017-07-01
- 虽然这个问题看起来很迷惑, 但从内部逻辑上来说呢, 其实是"对"的,我们来模拟下date内部的对于这种事情的处理逻辑:
- 先做-1 month, 那么当前是07-31, 减去一以后就是06-31.
- 再做日期规范化, 因为6月没有31号, 所以就好像2点60等于3点一样, 6月31就等于了7月1
- 也就是说, 只要涉及到大小月的最后一天, 都可能会有这个迷惑,我们印证一下这个逻辑:
var_dump(date("Y-m-d", strtotime("-1 month", strtotime("2017-03-31"))));
//输出2017-03-03
var_dump(date("Y-m-d", strtotime("+1 month", strtotime("2017-08-31"))));
//输出2017-10-01
var_dump(date("Y-m-d", strtotime("next month", strtotime("2017-01-31"))));
//输出2017-03-03
var_dump(date("Y-m-d", strtotime("last month", strtotime("2017-03-31"))));
//输出2017-03-03
怎么从坑里爬出来呢?
- 既然我们知道了“坑”的逻辑,那我们应该怎么爬出来呢?
- 从PHP5.3开始呢, date新增了一系列修正短语, 来明确这个问题, 那就是"first day of" 和 “last day of”, 也就是你可以限定好不要让date自动"规范化":
var_dump(date("Y-m-d", strtotime("last day of -1 month", strtotime("2017-03-31"))));
//输出2017-02-28
var_dump(date("Y-m-d", strtotime("first day of +1 month", strtotime("2017-08-31"))));
输出2017-09-01
var_dump(date("Y-m-d", strtotime("first day of next month", strtotime("2017-01-31"))));
输出2017-02-01
var_dump(date("Y-m-d", strtotime("last day of last month", strtotime("2017-03-31"))));
输出2017-02-28
- 现在, 搞清楚了内部原理, 也有了解决方法,是不是就轻松出坑了?
那耄耋老人,怎么出坑呢?
- 这么大岁数了,还在写代码呢?
- 那如果是5.3之前的版本(还有人用么?), 你可以使用mktime之类的, 把所有的日子忽略掉, 比如都限定为每月1号就可以了, 只不过就不如直接用first day来的更加优雅.