PHPでオブザーバーパターンを用いたイベント駆動システムの構築方法を徹底解説

イベント駆動型のシステムは、ユーザーの行動やシステムの変化に反応して動作する、柔軟かつ動的なシステム設計です。PHPでオブザーバーパターンを使用することで、イベント駆動型のシステムを効率的に構築できます。オブザーバーパターンは、あるオブジェクト(イベントの発生源)に対して複数のオブジェクト(オブザーバー)が監視し、イベントが発生した際に通知を受けて処理を行う仕組みです。

本記事では、PHPでオブザーバーパターンを用いたイベント駆動型システムの構築方法について、基本から実践例、さらに高度な応用までを解説します。

目次

オブザーバーパターンの基本概念


オブザーバーパターンは、あるオブジェクト(サブジェクト)が状態を変更した際に、それを監視している他のオブジェクト(オブザーバー)に自動的に通知を送るデザインパターンです。このパターンは、多対一の依存関係を構築し、サブジェクトとオブザーバー間の結合度を低く保つために使われます。

オブザーバーパターンの構成要素

  • サブジェクト:イベントや状態を発生させるオブジェクト。複数のオブザーバーを登録・解除し、変更時に通知を送信します。
  • オブザーバー:サブジェクトの状態変化を監視し、その通知を受けて動作を開始するオブジェクトです。

適用される状況


オブザーバーパターンは、イベント駆動型の通知機能を実装する場面で活用されます。例えば、ユーザーの操作に応じて通知を送信するシステムや、状態の変化を複数のコンポーネントに伝播させる必要があるシステムに適しています。

PHPにおけるオブザーバーパターンの実装


PHPでオブザーバーパターンを実装するには、まずサブジェクトとオブザーバーのインターフェースを定義し、それに基づいてクラスを作成します。これにより、サブジェクトからオブザーバーへの通知が簡単に行えるようになります。

サブジェクトとオブザーバーのインターフェース


まず、サブジェクト(イベントの発生源)とオブザーバー(通知を受け取る側)のインターフェースを作成します。

interface Subject {
    public function attach(Observer $observer);
    public function detach(Observer $observer);
    public function notify();
}

interface Observer {
    public function update();
}
  • attach: オブザーバーを登録します。
  • detach: 登録されたオブザーバーを解除します。
  • notify: 状態が変化した際に、全オブザーバーに通知を送ります。
  • update: オブザーバー側で通知を受け取った後の処理を行います。

具体的な実装例


次に、これらのインターフェースを使って具体的なクラスを作成します。

class EventSubject implements Subject {
    private $observers = [];

    public function attach(Observer $observer) {
        $this->observers[] = $observer;
    }

    public function detach(Observer $observer) {
        $this->observers = array_filter($this->observers, function($o) use ($observer) {
            return $o !== $observer;
        });
    }

    public function notify() {
        foreach ($this->observers as $observer) {
            $observer->update();
        }
    }

    public function triggerEvent() {
        // 何らかのイベントが発生したと仮定
        echo "イベントが発生しました!通知を送ります。<br>";
        $this->notify();
    }
}

class EventObserver implements Observer {
    private $name;

    public function __construct($name) {
        $this->name = $name;
    }

    public function update() {
        echo "{$this->name} が通知を受け取りました。<br>";
    }
}

実行例


上記のクラスを使用して、イベントが発生した際に複数のオブザーバーに通知を送る処理を行います。

$subject = new EventSubject();
$observer1 = new EventObserver("オブザーバー1");
$observer2 = new EventObserver("オブザーバー2");

$subject->attach($observer1);
$subject->attach($observer2);

$subject->triggerEvent();

この実行により、triggerEventメソッドが呼び出されたとき、登録されたオブザーバーに通知が送られ、それぞれが反応します。

イベント駆動型システムの概要


イベント駆動型システムとは、アプリケーション内で「イベント」と呼ばれる特定の出来事が発生したときに、事前に定義されたアクションが実行される仕組みを持つシステムです。イベント駆動型システムはユーザー操作や外部システムの状態変化に応答するため、リアルタイム性や柔軟な反応が求められる場面でよく利用されます。

イベント駆動型システムの特徴

  • 反応型アーキテクチャ:イベント駆動型システムは、システム内で発生したイベントに応じて動的に処理を行います。
  • 非同期処理:イベントが非同期で処理されることが多く、ユーザーに対して即時応答を可能にします。
  • モジュール性の向上:各コンポーネントが独立して動作し、互いに依存せずにイベントに応答するため、システムの拡張やメンテナンスが容易です。

オブザーバーパターンとの関係


オブザーバーパターンはイベント駆動型システムの一部として機能し、イベントを受け取る側の構成要素を実装するために利用されます。具体的には、イベントの発生元であるサブジェクトが状態の変化やイベントの発生を通知し、複数のオブザーバーがその通知を受けて反応します。これにより、サブジェクトとオブザーバーが疎結合となり、システムの柔軟性と保守性が向上します。

イベントハンドラの設定と実装


イベントハンドラは、イベントが発生した際に特定の処理を行うために設定される関数やメソッドのことです。PHPでイベント駆動型のシステムを実現する際、イベントハンドラを適切に設定することで、イベントが発生した際の反応をスムーズに処理できます。

イベントハンドラの役割


イベントハンドラは、以下のような役割を果たします。

  • イベントの検知:特定のイベントが発生したことを検知し、イベントに応じた処理を実行します。
  • 動的な対応:イベントの種類に応じて異なる処理を行い、システムの柔軟性を確保します。
  • アクションの管理:イベントが発生するたびに実行するアクションを効率よく管理するために、ハンドラでの処理を一元化します。

PHPでのイベントハンドラの実装方法


PHPでは、オブザーバーパターンを用いてイベントハンドラを設置できます。以下は、イベントが発生した際に通知を受けて処理を行うイベントハンドラのサンプルです。

class EventSubject {
    private $observers = [];

    public function attach(Observer $observer) {
        $this->observers[] = $observer;
    }

    public function detach(Observer $observer) {
        $this->observers = array_filter($this->observers, function($o) use ($observer) {
            return $o !== $observer;
        });
    }

    public function notify() {
        foreach ($this->observers as $observer) {
            $observer->handleEvent();
        }
    }

    public function triggerEvent() {
        echo "イベントが発生しました。ハンドラが起動します。<br>";
        $this->notify();
    }
}

class EventObserver implements Observer {
    public function handleEvent() {
        echo "イベントハンドラが実行されました。<br>";
    }
}

イベントハンドラの設定方法

  1. ハンドラの登録:オブザーバーに該当するクラスで、イベントが発生した際の処理をhandleEventメソッド内で定義します。
  2. イベントのトリガーtriggerEventメソッドでイベントが発生すると、notifyメソッドを介して全オブザーバーに通知が送られ、ハンドラが実行されます。

イベントの反応例

$subject = new EventSubject();
$observer = new EventObserver();

$subject->attach($observer);
$subject->triggerEvent();

このように、イベントハンドラを設置することで、イベントの発生に応じた処理を簡単に管理でき、柔軟なイベント駆動型システムの構築が可能です。

実際のイベント駆動型システムの流れ


イベント駆動型システムでは、イベントが発生した際に、それを起点として一連の処理が実行される流れが構築されます。この流れを理解することで、システム内でイベントがどのように処理され、オブザーバーパターンがどのように役立つかを把握できます。

1. イベントの発生


まず、システム内の特定のアクション(例:ユーザー操作、データ変更、タイマー発動など)によってイベントが発生します。このイベントは、サブジェクトであるイベント発生源によって管理されます。

イベントのトリガー例


サブジェクトのtriggerEventメソッドが呼び出され、イベントが発生します。

$subject->triggerEvent();

2. オブザーバーへの通知


イベントが発生すると、サブジェクトは事前に登録されたオブザーバー(監視者)に対して通知を送信します。サブジェクトとオブザーバーは、オブザーバーパターンによって疎結合に保たれているため、オブザーバーの追加や変更が容易です。

通知の仕組み


notifyメソッドを使用して、登録されているすべてのオブザーバーに通知を送ります。

foreach ($this->observers as $observer) {
    $observer->update();
}

3. オブザーバーによる処理の実行


各オブザーバーは通知を受け取ると、自身のupdateメソッドが実行され、必要な処理を行います。例えば、特定の画面にメッセージを表示する、データを更新する、あるいは他のサービスと連携するなどの処理を行います。

オブザーバーの動作例

class EventObserver implements Observer {
    public function update() {
        echo "オブザーバーがイベントに応答しました。<br>";
    }
}

4. 結果の確認・処理の完了


すべてのオブザーバーがイベントに応答し、処理が完了すると、システムは元の待機状態に戻ります。次のイベントが発生するまで待機し、再びイベントが発生すると同様の流れが繰り返されます。

流れのまとめ


イベントの発生 ➔ オブザーバーへの通知 ➔ 各オブザーバーの処理実行 ➔ 処理完了

この流れにより、イベントがシステム全体に伝播され、必要な応答を取ることが可能となります。PHPのオブザーバーパターンを用いることで、イベント駆動型システムの設計がシンプルかつ柔軟に行えます。

実践:簡易的な通知システムの構築


ここでは、PHPのオブザーバーパターンを利用して簡易的な通知システムを構築する方法を紹介します。この例を通じて、イベント駆動型システムの実装に必要な基本的な概念や実装方法を理解できます。

システムの概要


この通知システムでは、以下のようなシナリオを想定します:

  • システム内でイベントが発生すると、複数のユーザーに通知が送られる。
  • 各ユーザーは、通知を受け取ると独自のメッセージを表示する。

コードの実装


以下のコードでは、通知システムのサブジェクトとオブザーバーを作成します。サブジェクトがイベントをトリガーし、各オブザーバー(ユーザー)に通知を送ります。

// サブジェクト(通知システム)
class NotificationSystem {
    private $observers = [];

    // オブザーバーを追加
    public function addObserver(Observer $observer) {
        $this->observers[] = $observer;
    }

    // オブザーバーを解除
    public function removeObserver(Observer $observer) {
        $this->observers = array_filter($this->observers, function($o) use ($observer) {
            return $o !== $observer;
        });
    }

    // 通知を実行
    public function notifyObservers($message) {
        foreach ($this->observers as $observer) {
            $observer->receiveNotification($message);
        }
    }

    // イベントをトリガーするメソッド
    public function triggerEvent($message) {
        echo "イベントが発生しました。通知を送信します。<br>";
        $this->notifyObservers($message);
    }
}

// オブザーバー(ユーザー)
class User implements Observer {
    private $name;

    public function __construct($name) {
        $this->name = $name;
    }

    // 通知を受け取るメソッド
    public function receiveNotification($message) {
        echo "{$this->name} さんが通知を受け取りました:{$message}<br>";
    }
}

システムの利用例


システムに複数のユーザーを追加し、イベントが発生した際に各ユーザーが通知を受け取る流れを確認します。

// サブジェクトを作成
$notificationSystem = new NotificationSystem();

// ユーザー(オブザーバー)を作成して登録
$user1 = new User("ユーザー1");
$user2 = new User("ユーザー2");

$notificationSystem->addObserver($user1);
$notificationSystem->addObserver($user2);

// イベント発生と通知送信
$notificationSystem->triggerEvent("新しいイベントが発生しました!");

実行結果


実行すると、以下のような結果が表示されます:

イベントが発生しました。通知を送信します。
ユーザー1 さんが通知を受け取りました:新しいイベントが発生しました!
ユーザー2 さんが通知を受け取りました:新しいイベントが発生しました!

この例では、イベントが発生すると、登録されたユーザー(オブザーバー)にメッセージが通知され、それぞれが応答を返します。このように、オブザーバーパターンを活用することで、シンプルで柔軟な通知システムを構築することができます。

パフォーマンス向上のための工夫


イベント駆動型システムでは、効率的にイベントを処理するための工夫が求められます。特に、通知を受け取るオブザーバーが増加する場合、システムのパフォーマンスに影響を与える可能性があるため、適切な設計や最適化が必要です。

1. イベントのバッチ処理


頻繁に発生するイベントを一度にまとめて処理する「バッチ処理」は、パフォーマンス向上に効果的です。これにより、通知の頻度を抑え、CPUの負荷を軽減できます。

バッチ処理の例

class BatchedNotificationSystem extends NotificationSystem {
    private $eventQueue = [];

    public function queueEvent($message) {
        $this->eventQueue[] = $message;
    }

    public function processQueue() {
        foreach ($this->eventQueue as $message) {
            $this->notifyObservers($message);
        }
        $this->eventQueue = []; // 処理後はキューをクリア
    }
}

この例では、イベントを一旦キューに追加し、processQueueメソッドを呼び出すことで一括通知します。

2. 必要なオブザーバーのみへの通知


すべてのオブザーバーが同じイベントを必要とするわけではないため、条件に基づいて特定のオブザーバーのみ通知することもパフォーマンス向上に寄与します。例えば、通知にフィルタリングを追加し、必要なオブザーバーだけが通知を受け取るように設定できます。

特定の条件に基づく通知

public function notifyObserversWithCondition($message, $condition) {
    foreach ($this->observers as $observer) {
        if ($observer->shouldReceiveNotification($condition)) {
            $observer->receiveNotification($message);
        }
    }
}

3. メモリ使用量の削減


オブザーバーの数が増えると、メモリの使用量も増加するため、効率的なメモリ管理が重要です。不要なオブザーバーは早めに解除し、メモリの浪費を防ぎます。また、PHPでは大規模システムにおいてキャッシュやデータベースを活用し、メモリ使用量を抑える工夫も有効です。

オブザーバーの削除

$notificationSystem->removeObserver($observerToRemove);

4. 非同期処理の導入


非同期処理を用いることで、メインスレッドとは別にイベントの処理が行えるため、システムの応答性が向上します。PHPの非同期ライブラリ(ReactPHPやAmpなど)を活用することで、イベント処理を並行して行うことが可能です。

5. イベントログの活用によるデバッグと最適化


イベントの発生頻度や処理時間をログとして記録し、パフォーマンスのボトルネックを特定することで、システムの効率をさらに向上できます。ログから得られたデータをもとに、通知の間隔を調整したり、不要なイベントの除去を行うことができます。

これらの工夫を取り入れることで、イベント駆動型システムのパフォーマンスが向上し、大規模で多様なオブザーバーを持つシステムでもスムーズに稼働させることができます。

PHPフレームワークを活用した高度な実装例


PHPのフレームワークを使用すると、オブザーバーパターンを活用したイベント駆動型システムをより効率的に構築できます。特にLaravelのようなモダンなフレームワークは、イベントとリスナー(オブザーバー)機能を標準で提供しており、システム内で発生するさまざまなイベントに簡単に反応することができます。

Laravelにおけるイベントとリスナーの設定


Laravelでは、イベントとリスナーの設定が非常にシンプルです。php artisanコマンドを使って、イベントとリスナーを作成し、必要な処理を自動的に紐付けることができます。

イベントの作成


Laravelでは、次のコマンドでイベントを作成します。

php artisan make:event UserRegistered

上記のコマンドにより、UserRegisteredというイベントクラスが作成され、ユーザーが登録されたときに発生するイベントとして設定できます。

リスナーの作成


次に、リスナー(オブザーバー)を作成します。リスナーは、イベントが発生した際に実行される処理を定義する場所です。

php artisan make:listener SendWelcomeEmail

SendWelcomeEmailリスナーを作成し、このリスナーがUserRegisteredイベントを受け取るように設定します。

イベントとリスナーの関連付け


作成したイベントとリスナーを関連付けるには、EventServiceProviderに登録します。

protected $listen = [
    UserRegistered::class => [
        SendWelcomeEmail::class,
    ],
];

この設定により、UserRegisteredイベントが発生した際に、SendWelcomeEmailリスナーが自動的に実行されます。

コードの例:イベント発火とリスナーによる処理


実際にイベントを発火するには、次のようにしてUserRegisteredイベントを呼び出します。

use App\Events\UserRegistered;

// ユーザー登録後にイベントを発火
event(new UserRegistered($user));

このコードによって、UserRegisteredイベントが発生し、SendWelcomeEmailリスナーがその通知を受け取り、ユーザーにウェルカムメールを送信します。

Laravelイベントシステムのメリット

  • コードの分離:イベントとリスナーの分離により、各処理が明確に分かれ、保守性が向上します。
  • 再利用性:同じイベントに複数のリスナーを紐付けることができるため、柔軟な処理が可能です。
  • 非同期処理:キュー機能を活用することで、非同期に処理を行うことができ、システム全体のレスポンスを向上させられます。

LaravelをはじめとするPHPフレームワークを活用すると、オブザーバーパターンをより効率的かつ高度に実装することが可能になり、複雑なイベント駆動型システムの構築も簡単になります。

デバッグとトラブルシューティング


オブザーバーパターンを使ったイベント駆動型システムでは、デバッグやエラーの発生源を特定するためのトラブルシューティングが重要です。特に複数のオブザーバーが存在する場合、イベント通知の失敗や実行タイミングのズレなどがシステムの正常な動作に影響を与えることがあります。

1. イベント通知が機能しない場合の確認方法


イベント通知がオブザーバーに届かない場合、以下の点を確認します。

  • オブザーバーの登録漏れ:サブジェクトに対してオブザーバーを正しくattachメソッドで登録しているか確認します。
  • 通知メソッドの確認notifyメソッドが正しく呼び出され、各オブザーバーに通知が送信されているかを確認します。

デバッグ例


PHPでは、var_dump()print_r()を使ってオブザーバーの状態や通知メッセージを出力することができます。

var_dump($this->observers); // 登録されているオブザーバーを確認

2. オブザーバーの応答が遅い場合の対応


通知を受けたオブザーバーが処理に時間を要する場合、システム全体のパフォーマンスに影響が出る可能性があります。この場合、非同期処理やキューを使って通知を遅延させることで解決できることがあります。

  • 非同期処理:イベントを非同期で処理することで、他のオブザーバーが並行して実行されるようにします。
  • キュー処理:通知を一旦キューに入れて、順次処理することで負荷を分散します。

3. 依存関係の管理


イベント駆動型システムでは、オブザーバー間の依存関係が発生しやすくなります。例えば、あるオブザーバーの処理結果に応じて次のオブザーバーが動作する場合、依存関係を明確にする必要があります。これにより、デバッグ時にどのオブザーバーがどのイベントに応答しているのかが分かりやすくなります。

4. ログを活用したトラブルシューティング


各イベントの発生とオブザーバーの応答をログとして記録することで、デバッグとトラブルシューティングが容易になります。PHPでは、error_log()関数やMonologなどのロギングライブラリを使って、イベントの流れを追跡できます。

ログ出力例

error_log("イベントが発生しました:{$eventName}");
error_log("{$observerName} が通知を受け取りました");

5. Laravelにおけるイベントデバッグ


Laravelのようなフレームワークを使っている場合、イベントのリスナーの実行状況やキューの処理状況を専用のコマンドで確認できます。

  • イベントの監視php artisan event:listenコマンドで、イベントが発生した際の状況をリアルタイムで確認できます。
  • ログの活用:Laravelのロギング機能を活用して、各リスナーが正しく動作しているかを確認します。

これらのデバッグ方法を取り入れることで、オブザーバーパターンを用いたイベント駆動型システムの安定性と信頼性を高めることが可能です。デバッグの手法を活用しながら、システムの各部分が正しく連携して動作しているかを確実に確認しましょう。

応用編:複数イベントに対応する方法


システムが複雑になると、複数の異なるイベントが同時に発生し、それぞれに対応するオブザーバーが必要になります。この章では、複数のイベントに柔軟に対応するための設計方法と実装例を紹介します。

イベントタイプごとのオブザーバー登録


異なるイベントに対応するため、イベントタイプごとにオブザーバーを登録する方法があります。例えば、ユーザー登録イベントと注文完了イベントで異なる通知が必要な場合、イベントタイプごとにオブザーバーを整理することで、不要な通知を避けることができます。

イベントタイプの設定

class MultiEventSubject {
    private $observers = [];

    public function attach($eventType, Observer $observer) {
        $this->observers[$eventType][] = $observer;
    }

    public function notify($eventType) {
        if (isset($this->observers[$eventType])) {
            foreach ($this->observers[$eventType] as $observer) {
                $observer->update();
            }
        }
    }

    public function triggerEvent($eventType) {
        echo "{$eventType} イベントが発生しました。通知を送信します。<br>";
        $this->notify($eventType);
    }
}

このコードでは、イベントタイプごとにオブザーバーを登録し、指定されたタイプのイベントが発生したときのみ該当するオブザーバーに通知が送られるようにしています。

複数イベントに対応するオブザーバー


1つのオブザーバーが複数のイベントに対応できるようにする方法もあります。オブザーバーが対応するイベントタイプを複数保持し、各イベントごとに異なる処理を行えるようにします。

複数イベント対応のオブザーバー

class MultiEventObserver implements Observer {
    private $name;
    private $events = [];

    public function __construct($name, array $events) {
        $this->name = $name;
        $this->events = $events;
    }

    public function update($eventType) {
        if (in_array($eventType, $this->events)) {
            echo "{$this->name} が {$eventType} イベントを受け取りました。<br>";
        }
    }
}

オブザーバーは対応するイベントの種類を配列で持ち、通知時に対応するイベントのみ処理するようにしています。

実行例


実際にイベントの発生に応じて、オブザーバーに通知を行います。

$subject = new MultiEventSubject();

$userObserver = new MultiEventObserver("ユーザー通知", ["UserRegistered", "UserDeleted"]);
$orderObserver = new MultiEventObserver("注文通知", ["OrderPlaced"]);

$subject->attach("UserRegistered", $userObserver);
$subject->attach("UserDeleted", $userObserver);
$subject->attach("OrderPlaced", $orderObserver);

$subject->triggerEvent("UserRegistered");
$subject->triggerEvent("OrderPlaced");

柔軟な拡張性の実現


この方法を用いることで、複数のイベントに対応するシステムを柔軟に設計でき、イベントごとに必要なオブザーバーのみが通知を受け取る構造が確立されます。また、新しいイベントやオブザーバーの追加が容易になり、システムの拡張性が向上します。

このように複数イベントに対応する設計を取り入れることで、イベント駆動型システムの幅広い応用が可能となり、複雑なシステムでも効率的に管理できるようになります。

オブザーバーパターンの利点と限界


オブザーバーパターンを使うことで、イベント駆動型システムにおいて柔軟で拡張性のある設計を実現できますが、適用には一定の限界や考慮すべき課題もあります。ここでは、オブザーバーパターンのメリットと限界について解説します。

オブザーバーパターンの利点

  1. 柔軟性と拡張性の向上
    サブジェクトとオブザーバーが疎結合になるため、新しいオブザーバーを追加してもサブジェクトの実装に影響を与えません。この構造により、システムを柔軟に拡張できます。
  2. 再利用性
    オブザーバーは特定のイベントに対する処理を独立して保持するため、同じオブザーバーを別のサブジェクトやイベントに再利用することが容易です。
  3. メンテナンス性の向上
    イベント発生と通知処理が分離されているため、各オブザーバーやサブジェクトの変更が他の要素に与える影響が少なく、メンテナンスが容易になります。

オブザーバーパターンの限界

  1. 複雑な依存関係の発生
    オブザーバーが増えると、各オブザーバー間の依存関係が複雑になり、トラブルシューティングやデバッグが難しくなる可能性があります。また、イベントの通知タイミングや順序が複雑になると、予期しない動作が発生しやすくなります。
  2. パフォーマンスへの影響
    大量のオブザーバーが登録されている場合、一度に通知が送られることでシステムのパフォーマンスに影響が出る可能性があります。パフォーマンスを最適化するには、非同期処理やキュー処理の導入が必要になることもあります。
  3. デバッグの難しさ
    通知が非同期で行われる場合や、イベントが複雑に絡み合っている場合、デバッグやエラーの原因追跡が困難になることがあります。ログを活用した管理や、デバッグツールの利用が重要です。

適用が難しいケース


オブザーバーパターンは、リアルタイムの応答性や柔軟な通知が求められる場面で有効ですが、すべてのシステムに適用できるわけではありません。例えば、厳密な同期性が求められるシステムや、パフォーマンス重視のリアルタイム処理システムには別の設計パターンが適している場合があります。

オブザーバーパターンの特性を理解し、その利点と限界を把握することで、最適なシステム設計が可能になります。

まとめ


本記事では、PHPでオブザーバーパターンを利用したイベント駆動型システムの構築方法について解説しました。オブザーバーパターンの基本概念から、実装方法、イベント駆動システムでの役割、パフォーマンス向上の工夫、高度なフレームワーク活用までを網羅しました。

オブザーバーパターンを活用することで、柔軟かつ拡張性の高いシステムを実現できますが、依存関係の複雑さやパフォーマンスの課題もあります。適切なデバッグ手法や非同期処理を取り入れ、実際のニーズに合った構造にすることが重要です。オブザーバーパターンの利点を生かして、効率的なイベント駆動型システムを構築しましょう。

コメント

コメントする

目次