現代の Web アプリケーション開発では、データを効率的かつ安全に管理および転送することが重要です。このプロセスを大幅に支援する設計パターンの 1 つは、データ転送オブジェクト (DTO) です。この投稿では、特に Laravel アプリケーションで DTO を使用する利点を詳しく掘り下げ、PHP 8.2 読み取り専用クラスがその利点をさらに高める方法を示します。
データ転送オブジェクト (DTO) は、プロセスまたはシステム間でデータを転送するために設計された単純なオブジェクトです。一般的なモデルやエンティティとは異なり、DTO にはビジネス ロジックがありません。データをカプセル化し、アプリケーションの異なる層間またはさまざまなシステム間で情報を転送するための明確で構造化された方法を提供します。
DTO パターンは、ソフトウェア アプリケーション内のさまざまなサブシステム間でデータを転送するために利用されます。 DTO を使用する主な目的は、メソッド呼び出しの数を最小限に抑え、必要なデータを集約し、データ変換と検証を管理するための構造化されたアプローチを提供することです。
懸念事項の分離: DTO はビジネス ロジックをデータ表現から分離し、その結果、コードがよりクリーンで保守しやすく、理解しやすくなります。
データ検証: DTO により、他のアプリケーション層によって処理される前にデータを検証でき、有効なデータのみが使用されることが保証されます。
一貫性: データ転送に一貫した構造を提供することで、DTO はさまざまなソースからのデータの管理と処理を簡素化します。
セキュリティ: DTO は、アクセスおよび変更可能なデータを制御することで、アプリケーションを不正なデータ操作から保護できます。
テスト: DTO はビジネス ロジックが組み込まれていない単純なオブジェクトであるため、モックとテストが簡単です。
変換: DTO は、さまざまなアプリケーション層で必要な形式へのデータの変換を容易にします。
不変性: DTO は多くの場合、不変性を促進します。これは、一度作成されると状態を変更できないことを意味します。この機能にはいくつかの利点があります:
PHP 8.2 では、読み取り専用クラスの導入により DTO の使用が強化されています。読み取り専用クラスを使用すると、プロパティを読み取り専用として明示的に定義する必要がなくなり、DTO の実装が簡素化されます。 PHP 8.2 の読み取り専用クラスが DTO を改善する方法は次のとおりです。
API や CSV インポートなどのさまざまなソースからプロパティを取得できるプロパティ管理システムを考えてみましょう。 DTO を使用してプロパティ モデル、サブスクリプション、アセットなどを作成し、アプリケーション全体でデータの一貫性と検証を確保できます。
まず、PropertyDTO クラスを定義しましょう:
app/DTO/PropertyDTO.php
namespace App\DTO; /** * Class PropertyDTO * * Represents a Data Transfer Object for property data. */ readonly class PropertyDTO extends AbstractDTO { /** * The name of the property. * * @var string */ public string $name; /** * The address of the property. * * @var string */ public string $address; /** * The price of the property. * * @var float */ public float $price; /** * The subscription status of the property, if applicable. * * @var string|null */ public ?string $subscription; /** * The list of assets associated with the property. * * @var array|null */ public ?array $assets; /** * Set the properties from a model instance. * * @param $model The model instance. * @return $this */ public function setFromModel($model): self { $this->name = $model->name; $this->address = $model->address; $this->price = $model->price; $this->subscription = $model->subscription; $this->assets = $model->assets; return $this; } /** * Set the properties from API data. * * @param array $data The API data. * @return $this */ public function setFromAPI(array $data): self { $this->name = $data['property_name']; $this->address = $data['property_address']; $this->price = $data['property_price']; $this->subscription = $data['subscription'] ?? null; $this->assets = $data['assets'] ?? null; return $this; } /** * Set the properties from CSV data. * * @param array $data The CSV data. * @return $this */ public function setFromCSV(array $data): self { $this->name = $data[0]; $this->address = $data[1]; $this->price = (float) $data[2]; $this->subscription = $data[3] ?? null; $this->assets = explode(',', $data[4] ?? ''); return $this; } }
PropertyDTO を使用してさまざまなソースからのプロパティを処理する方法は次のとおりです。
// From a Model $model = Property::find(1); $propertyDTO = (new PropertyDTO([]))->setFromModel($model); // From an API $apiData = [ 'property_name' => 'Beautiful House', 'property_address' => '1234 Elm Street', 'property_price' => 450000, 'subscription' => 'Premium', 'assets' => ['pool', 'garden'] ]; $propertyDTO = (new PropertyDTO([]))->setFromAPI($apiData); // From a CSV $csvData = ['Beautiful House', '1234 Elm Street', 450000, 'Premium', 'pool,garden']; $propertyDTO = (new PropertyDTO([]))->setFromCSV($csvData); // Convert to Array $arrayData = $propertyDTO->toArray(); // Convert to JSON $jsonData = $propertyDTO->toJson();
データ転送オブジェクト (DTO) は、データの一貫性、検証、懸念事項の分離を保証することで、Laravel アプリケーションに多くの利点をもたらします。 DTO を実装すると、アプリケーションの保守性、安全性が向上し、テストが容易になります。プロパティ管理システムでは、DTO は API や CSV インポートなどのさまざまなソースからのデータを効率的に処理するのに役立ち、ビジネス ロジックがクリーンな状態に保たれ、検証済みデータの処理に集中できるようになります。
Moreover, embracing immutability within DTOs enhances predictability, thread-safety, and simplifies debugging.
To streamline the creation of DTOs and promote code reuse, we can use an abstract class or base class. This approach allows us to define common methods and properties in the abstract class and extend it for specific data sources.
app/DTO/AbstractDTO.php
namespace App\DTO; /** * AbstractDTO * * An abstract base class for Data Transfer Objects (DTOs). * Provides common methods and properties for DTO implementations. */ abstract class AbstractDTO { /** * AbstractDTO constructor. * * Initialises the DTO with data from an associative array. * * @param array $data The data array to initialize the DTO. */ public function __construct(array $data) { $this->setFromArray($data); } /** * Set the properties of the DTO from a model instance. * * @param $model The model instance from which to populate the DTO. * @return $this */ abstract public function setFromModel($model): self; /** * Set the properties of the DTO from API data. * * @param array $data The data array from the API. * @return $this */ abstract public function setFromAPI(array $data): self; /** * Set the properties of the DTO from CSV data. * * @param array $data The data array from the CSV. * @return $this */ abstract public function setFromCSV(array $data): self; /** * Convert the DTO to an associative array. * * @return array The DTO data as an associative array. */ public function toArray(): array { $properties = get_object_vars($this); return array_filter($properties, function ($property) { return $property !== null; }); } /** * Convert the DTO to a JSON string. * * @return string The DTO data as a JSON string. */ public function toJson(): string { return json_encode($this->toArray()); } /** * Set the properties of the DTO from an associative array. * * @param array $data The data array to populate the DTO. */ protected function setFromArray(array $data): void { foreach ($data as $key => $value) { if (property_exists($this, $key)) { $this->$key = $value; } } } }
Using an abstract or base class for DTOs not only ensures consistency across different DTO implementations but also promotes code reuse and maintainability. By defining common methods and properties in an abstract class, you can create a structured and efficient way to manage data transfer within your application. This approach aligns well with the principles of clean code and helps in building scalable and robust applications.
Here’s a revised phrase that includes a call to action:
"By leveraging DTOs and abstract classes together, you can refine your Laravel application's design, improving how data is managed and ensuring a more organised and efficient data flow. If you want to further encapsulate and enhance your DTOs with traits and interfaces, explore our guide on Enhancing Object-Oriented Design with Traits, Interfaces, and Abstract Classes."
以上がデータ転送オブジェクト (DTO) の利点と、PHP 読み取り専用クラスが Laravel コードをどのように昇格できるかを探るの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。