解決strtotime增加減少一個月日期不準的問題

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

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

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

  1. 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

是不是邏輯很”清晰”呢? 我們也可以手動驗證第二個步驟, 比如:

  1. var_dump(date("Y-m-d", strtotime("2017-06-31")));
  2. //輸出2017-07-01

也就是說, 只要涉及到大小月的最後一天, 都可能會有這個迷惑, 我們也可以很輕鬆的驗證類似的其他月份, 印證這個結論:

  1. var_dump(date("Y-m-d", strtotime("-1 month", strtotime("2017-03-31"))));
  2. //輸出2017-03-03
  3. var_dump(date("Y-m-d", strtotime("+1 month", strtotime("2017-08-31"))));
  4. //輸出2017-10-01
  5. var_dump(date("Y-m-d", strtotime("next month", strtotime("2017-01-31"))));
  6. //輸出2017-03-03
  7. var_dump(date("Y-m-d", strtotime("last month", strtotime("2017-03-31"))));
  8. //輸出2017-03-03

那怎麼辦呢?

從PHP5.3開始呢, date新增了一系列修正短語, 來明確這個問題, 那就是”first day of” 和 “last day of”, 也就是你可以限定好不要讓date自動”規範化”:

  1. var_dump(date("Y-m-d", strtotime("last day of -1 month", strtotime("2017-03-31"))));
  2. //輸出2017-02-28
  3. var_dump(date("Y-m-d", strtotime("first day of +1 month", strtotime("2017-08-31"))));
  4. ////輸出2017-09-01
  5. var_dump(date("Y-m-d", strtotime("first day of next month", strtotime("2017-01-31"))));
  6. ////輸出2017-02-01
  7. var_dump(date("Y-m-d", strtotime("last day of last month", strtotime("2017-03-31"))));
  8. ////輸出2017-02-28

那如果是5.3之前的版本(還有人用麼?), 你可以使用mktime之類的, 把所有的日子忽略掉, 比如都限定爲每月1號就可以了, 只不過就不如直接用first day來的更加優雅.

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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章