令人困惑的strtotime

經常會有人被strtotime結合-1 month, +1 month, next month的時候搞得很困惑, 然後就會覺得這個函數有點不那麼靠譜, 動不動就出問題. 用的時候就會很慌…

這不, 剛剛就有人在微博上又問我:

鳥哥,今天是2018-07-31 執行代碼:

 date("Y-m-d",strtotime("-1 month"))

怎麼輸出是2018-07-01?

好的吧, 雖然這個問題看起來很迷惑, 但從內部邏輯上來說呢, 其實是”對”的, 你先別著急哈, 讓我慢慢講:

我們來模擬下date內部的對於這種事情的處理邏輯:

  • 1. 先做-1 month, 那麼當前是07-31, 減去一以後就是06-31.
  • 2. 再做日期規範化, 因為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來的更加優雅.

現在, 搞清楚了內部原理, 是不是就不慌了? 🙂

发表评论

电子邮件地址不会被公开。 必填项已用*标注