2
0

【Laravel】【PHP】トレイト(Trait)を活用して実装を再利用する

Last updated at Posted at 2024-08-31

はじめに

今回はPHPに実装されている
 トレイト(Trait)
と呼ばれる仕組みについて記載してみました。

トレイト(Trait)とは

以下、PHP公式ドキュメントでの説明が下記となります

PHP は、コードを再利用するための「トレイト」という仕組みを実装しています。

トレイトは、PHP のような単一継承言語でコードを再利用するための仕組みのひとつです。 トレイトは、単一継承の制約を減らすために作られたもので、 いくつかのメソッド群を異なるクラス階層にある独立したクラスで再利用できるようにします。 トレイトとクラスを組み合わせた構文は複雑さを軽減させてくれ、 多重継承や Mixin に関連するありがちな問題を回避することもできます。

トレイトはクラスと似ていますが、トレイトは単にいくつかの機能をまとめるためだけのものです。 トレイト自身のインスタンスを作成することはできません。 昔ながらの継承に機能を加えて、振る舞いを水平方向で構成できるようになります。 つまり、継承しなくてもクラスのメンバーに追加できるようになります。

個人的な解釈

ここからは個人的な解釈となります。

トレイト(Trait)

関連のない【複数クラスで使用する可能性のあるメソッドを再利用できるようにする】ための仕組みです。

継承

【複数クラスで使用する可能性のあるメソッドを再利用できるようにする】を実現させるにあたって、トレイト(Trait)以外を使った方法を先に紹介します。

それが 継承 と呼ばれる仕組みを使った方法です。

継承 は超ざっくり言うと、親クラスの構造を引き継いだ(拡張した)子クラスを定義することです。継承した関係にあるクラスであれば基本的に親と同じメソッドを持っているので、メソッドの再利用を実現可能です。

class Hoge{

    protected string $name = 'hoge';

    public function toArray()
    {
        return get_object_vars($this);
    }
}

class HogeFuga extends Hoge{

    protected string $name = 'hoge-fuga';

}

$hogeFuga = new HogeFuga;
var_dump($hogeFuga->toArray());

// 出力
// array(1) {
//   ["name"]=>
//   string(8) "HogeFuga"
// }

上記コードのサンプルでは
Hogeクラスを継承したHogeFugaクラスを用意しました。

このHogeFugaクラス(子)はHogeクラス(親)を継承しているので
Hogeクラス(親)に搭載されているtoArrayメソッドを使用可能です。

親子関係のない別のクラスにもtoArrayを取り入れたい...

ある時、HogeクラスのtoArray()メソッドを
用途としてはまったく関係のない別のColorクラスにも導入したいとなりました。

この際にHogeを継承させたColorクラスを作るやり方だと
Hogeクラスに変更を加えた際にColorクラスまで影響を受けることになります。

今回はあくまでメソッドだけを取り入れたいので
トレイト(Trait) を使ってメソッドだけを取り入れることにしました。

<?php

trait toArrayConvertable{
    
    public function toArray()
    {
        return get_object_vars($this);
    }
}

class Hoge{
    
    use toArrayConvertable;

    protected string $name = 'hoge';

}

class HogeFuga extends Hoge{

    protected string $name = 'hoge-huga';

}


// 関係のないColorクラス
class Color{
      
    use toArrayConvertable;
    
    protected string $name = 'color';

}


$hogeFuga = new HogeFuga;
var_dump($hogeFuga->toArray());

$color = new Color;
var_dump($color->toArray());

// 出力
// array(1) {
//   ["name"]=>
//   string(9) "hoge-huga"
// }
// array(1) {
//   ["name"]=>
//   string(5) "color"
// }

?>

Traitを使用することで関係のないクラスに対して
メソッドを提供することができました。

トレイト(Trait)と継承の使い分け

個人的な使い分けの観点ですが

  • 親子関係にあるクラス内でのメソッド => 継承
  • 親子関係にないクラス間でのメソッド => トレイト(Trait)

というイメージで使い分けています。

ユースケース

トレイト(Trait)クラスの活用例をいくつか載せてみました。
Laravel内で使用されている例や、個人的にLaravelでの実装時に使用した事例となります。

Laravelで特定モデルへ論理削除を適用する

こちらはLaravelで使用されている例となります。

Laravelで特定のテーブルに論理削除を導入したい場合に、
モデルクラスへIlluminate\Database\Eloquent\SoftDeletesのトレイトを適用することで
簡単に論理削除を導入することができます。
deleted_atを基準として論理削除となります)

こちらはLaravel 11でインストール時に作られるUserモデルクラスへ
論理削除のトレイトを適用する例です。

app/Models/User.php
<?php

namespace App\Models;

// use Illuminate\Contracts\Auth\MustVerifyEmail;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;
use Illuminate\Database\Eloquent\SoftDeletes; //★ Use宣言

class User extends Authenticatable
{
    use HasFactory, Notifiable, SoftDeletes; //★ SoftDeletesの適用

    /**
     * The attributes that are mass assignable.
     *
     * @var array<int, string>
     */
    protected $fillable = [
        'name',
        'email',
        'password',
    ];

    /**
     * The attributes that should be hidden for serialization.
     *
     * @var array<int, string>
     */
    protected $hidden = [
        'password',
        'remember_token',
    ];

    /**
     * Get the attributes that should be cast.
     *
     * @return array<string, string>
     */
    protected function casts(): array
    {
        return [
            'email_verified_at' => 'datetime',
            'password' => 'hashed',
        ];
    }
}

特定機能のアクションメソッドを定義する

こちらは個人的に試した事例となります。

各リソースに対する機能のうち、
CSV形式で出力するルーティングexport/csvをコントローラに対して設定したケースです。

こちらのCSV出力機能はリソースによっては追加しない想定のため、
アクションメソッドをトレイトで共通化しました。

app/Http/Controllers/Controller.php
<?php

namespace App\Http\Controllers;

use App\Models\User;
use App\Http\Controllers\Traits\CsvExportable;

class UserController exuends Controller
{
    use CsvExportable;

    protected $model = User::class;
}
app/Http/Controllers/Traits/CsvExportable.php
<?php

namespace App\Http\Controllers\Traits;

use Illuminate\Http\Request;

trait CsvExportable
{
    /**
     * CSVエクスポート
     */
    public function exportCsv(Request $request)
    {
        if (is_null($this->model)) {
            throw new \Exception('modelプロパティが設定されていません');
        }

        // CSV出力処理
    }
}

まとめ

  • Trait(トレイト)は実装したメソッドを共通化するための仕組みの1つ
  • Traitを実装して、クラス内へuse トレイト名でメソッドが可能になる
<?php

trait Greetable {

    public function hello()
    {
        echo 'hello!';
    }
}

class Hoge {

    use Greetable; // Greetableトレイトのメソッドを使用可能とする
    
    public function hoge()
    {
        echo 'hoge';
    }
}

$hoge = new Hoge;
$hoge->hello(); // hogeが挨拶してくれたようです

// 出力
// hello!
?>

参考

告知

最後にお知らせとなりますが、イーディーエーでは一緒に働くエンジニアを
募集しております。詳しくは採用情報ページをご確認ください。

みなさまからのご応募をお待ちしております。

2
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
0