PHPで、先月、翌月などを扱うときの注意

※ かなり初歩的な話ですのでご了承ください


PHPで、先月、翌月の月を取得するような場合、以下のようなプログラムを書いてしまいませんか?

echo '先月 => '. date('m', strtotime('-1 month')) .'<br>';
echo '今月 => '. date('m') .'<br>';
echo '翌月 => '. date('m', strtotime('+1 month')) .'<br>';


------ 結果 ------

前月 => 10
今月 => 11
翌月 => 12


(今日を11月01日とした場合です)

上記の結果は期待した結果で現状11月01日の場合は問題ありません。


ところが、10月31日の場合を例にとって見ます。

echo '前月 => '. date('m', strtotime('2009-10-31 -1 month')) .'<br>';
echo '今月 => '. date('m', strtotime('2009-10-31') ) .'<br>';
echo '翌月 => '. date('m', strtotime('2009-10-31 +1 month')) .'<br>';


------ 結果 ------
先月 => 10
今月 => 10
翌月 => 12

大変なことに、先月が10になってしまいました。
もっと、過去にもどって見てみると

for($i=1;$i<=12;$i++){
  echo $i. 'ヶ月前 => '. date('m', strtotime('2009-10-31 -'.$i.' month')) .'<br>';
}

------ 結果 ------
1ヶ月前 => 10
2ヶ月前 => 08
3ヶ月前 => 07
4ヶ月前 => 07
5ヶ月前 => 05
6ヶ月前 => 05
7ヶ月前 => 03
8ヶ月前 => 03
9ヶ月前 => 01
10ヶ月前 => 12
11ヶ月前 => 12
12ヶ月前 => 10

もう、ありえないことになってしまいますね。
よく見ると31日のある日とない日で計算が狂ってしまうという内容です。
2月にいたっては、29〜31日もアウトですね。


これはおそらく10月31日の1ヶ月前は9月31日となり
9月31日とは9月30日の次の日だから10月1日のことになるってことでしょう。
そのため、10月31日の1ヶ月前の月は10月って事になります。


strtotime の -1 month とは、日付はそのままで月だけ1減らすという意味で
決して先月にシフトするという意味ではない。
関数を作った側の理屈はなんとなくあっている気がするのですが
使ってる側にしてみると間違いやすいポイントかなと思います。


従いまして、先月、翌月とかを正確に取得するには現在の日を考慮いれて
今月の1日から起算して1ヶ月前とかになるようにします。

echo '先月 => '. date('m', strtotime(date('Y-m-1').' -1 month')) .'<br>';
echo '今月 => '. date('m') .'<br>';
echo '翌月 => '. date('m', strtotime(date('Y-m-1').' +1 month')) .'<br>';

------ 結果 ------

前月 => 10
今月 => 11
翌月 => 12
for($i=1;$i<=12;$i++){
  echo $i. 'ヶ月前 => '. date('m', strtotime(date('Y-m-1').' -'.$i.' month')) .'<br>';
}

------ 結果 ------

1ヶ月前 => 10
2ヶ月前 => 09
3ヶ月前 => 08
4ヶ月前 => 07
5ヶ月前 => 06
6ヶ月前 => 05
7ヶ月前 => 04
8ヶ月前 => 03
9ヶ月前 => 02
10ヶ月前 => 01
11ヶ月前 => 12
12ヶ月前 => 11


ぶっちゃけ、昨日上記のバグを発見したからこのエントリーなんですけどね(汗)
かなり初歩的なバグが混入していて情けない思いをしました。未熟者です。
こんなのは常識だよーって言われそうですが、知らない人がいましたら後学のために参考になれば幸いです。