經常會有人被strtotime結合-1 month, +1 month, next month的時候搞得很困惑, 然后就會覺得這個函數有點不那么靠譜, 動不動就出問題. 用的時候就會很慌…
今天是2018-08-31 執行代碼:
date("Y-m-d",strtotime("+1 month"))
怎么輸出是2018-10-01?
雖然這個問題看起來很迷惑, 但從內部邏輯上來說呢, 其實是”對”的
date內部的對於這種事情的處理邏輯:
- 先做-1 month, 那么當前是07-31, 減去一以后就是06-31.
- 再做日期規范化, 因為6月沒有31號, 所以就好像2點60等於3點一樣, 6月31就等於了7月1
是不是邏輯很”清晰”呢? 我們也可以手動驗證第二個步驟, 比如:
var_dump(date("Y-m-d", strtotime("2017-06-31")));
//輸出2017-07-01
也就是說, 只要涉及到大小月的最后一天, 都可能會有這個迷惑, 我們也可以很輕松的驗證類似的其他月份, 印證這個結論:
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來的更加優雅.
有人將這個問題提交為bug ,https://bugs.php.net/bug.php?id=22486