🕰️

【PHP/Laravel】Carbonの日付計算メソッド - 単数形・複数形の違いについて

2024/11/20に公開

結論

  • 現状の最新バージョンにおいても「挙動」については違いはない。
  • ただし、公式としては以下のような使い分けを想定しているようである。
    • 単数系メソッド(例:addDay()/subYear()など):日付を「1つ分だけ」計算する時
    • 複数形メソッド(例:addDays()/subYears()など):「2つ分以上」計算する時
  • 利用する際はライブラリのアップデート時に予期せぬ不具合を起こさないように公式の想定に準拠しておいた方が良さそう(今後のアップデートで単数系メソッドに引数を渡せないようになる可能性も考えられるため)

環境

ソフトウェア バージョン
PHP 8.3.13
Laravel 11.32.0
Carbon 3.8.2

内容

PHPには標準DateTimeクラスを拡張したCarbonライブラリというものが存在します。
Laravelなどでは標準搭載されているので利用したことがある人も多いのではないでしょうか。

Carbonライブラリには日付を計算するメソッドとして単数系メソッド(例:addDay()/subYear()など)と複数形メソッド(例:addDays()/subYears()など)が用意されています。

具体的な違いについて理解するために、例えば「10月10日の3日後を取得したい」ケースを想定してみましょう。「日」の加算ではaddDayaddDaysの2種類を利用できます。

メソッドの命名からaddDayは「1日分だけ後ろの日付を取得できる(= 2日分以上後ろの日付を取得できない)」と仮定してみます。いざ確認してみましょう。

php artisan tinker

> $date = new Carbon\Carbon('2024-10-10');
= Carbon\Carbon @1728518400 {#5207
    date: 2024-10-10 00:00:00.0 UTC (+00:00),
  }

// 引数なしパターンを検証
> $date->addDay();
= Carbon\Carbon @1728604800 {#5207
    date: 2024-10-11 00:00:00.0 UTC (+00:00),
  }

> $date = new Carbon\Carbon('2024-10-10');
= Carbon\Carbon @1728518400 {#5188
    date: 2024-10-10 00:00:00.0 UTC (+00:00),
  }

// 引数ありパターンを検証
> $date->addDay(3);
= Carbon\Carbon @1728777600 {#5188
    date: 2024-10-13 00:00:00.0 UTC (+00:00),
  }

あれ?「1日分後ろの日付」に加えて、引数を渡して「3日分後ろの日付」も取得できました🤔

一旦、これ自体は頭の片隅に置いてもう一方のaddDaysメソッドを試してみましょう。

> $date = new Carbon\Carbon('2024-10-10');
= Carbon\Carbon @1728518400 {#5200
    date: 2024-10-10 00:00:00.0 UTC (+00:00),
  }

// 引数なしパターンを検証
> $date->addDays();
= Carbon\Carbon @1728604800 {#5198
    date: 2024-10-11 00:00:00.0 UTC (+00:00),
  }

> $date = new Carbon\Carbon('2024-10-10');
= Carbon\Carbon @1728518400 {#5188
    date: 2024-10-10 00:00:00.0 UTC (+00:00),
  }

// 引数ありパターンを検証
> $date->addDays(3);
= Carbon\Carbon @1728777600 {#5188
    date: 2024-10-13 00:00:00.0 UTC (+00:00),
  }

こちらは想像通りですね。
これらの結果から「挙動」についてCarbon 3.8.2バージョンについては違いがないことが分かります。

そこで「公式の見解」を見てみましょう。
まずは公式ドキュメントから確認してみます。

まずは複数形のaddDaysメソッドではint|float $value = 1と書いてありますね。納得できます。

次に単数系のaddDayメソッドを見てみるとno argumentsと記載されています。(困惑)

次に実際のソースコード(Carbon.php L390)を確認してみます。

// ※見やすさ観点でインデントを少なくしています。
 * @method $this addDays(int|float $value = 1) Add days (the $value count passed in) to the instance (using date interval).
 * @method $this addDay() Add one day to the instance (using date interval).

Add one day to the instanceとあるように単数系のaddDayメソッドでは「1日分のみ加算」することを想定していそうです。

また、テストコード(AddTest.php L327)でも以下のような記述が見つかりました。

public function testAddDayPassingArg()
{
    // addDay should ideally be used without argument
    /** @var mixed $date */
    $date = Carbon::createFromDate(1975, 5, 10);
    $this->assertSame(12, $date->addDay(2)->day);
}

「addDayメソッドは、引数なしで使うのが望ましい」と書かれていますね。

以上の確認から今回のようなケースの場合では「addDaysメソッドに引数を渡して利用すること」が公式の見解に則っていると言えるでしょう。(逆に「翌日の日付を取得したい場合」はaddDayメソッドを利用する)

今後のバージョンアップで仕様が変更される可能性もあるため、利用する際は公式の見解に従ってコーディングすることをお勧めします。

ここまでお読み頂き、ありがとうございました。

Discussion