メインメニュー
XAMPP アレンジ
IED
WSL2
- 
道具箱リポジトリ編フレームワーク編公開ソフトウェアメタリンク- PHP ライブラリ- PHP 言語
 apricot:ext:di-container目次文書の過去の版を表示しています。 
 編集中
 Apricot DIコンテナー— y2sunlight 2020-05-25 関連記事 
- Apricot 拡張
- Apricot DIコンテナー
 ApricotではDIコンテナーにLeague/Containerを採用します。主な用途はコントローラに対する Auto Wiringです。Auto Wiringとはコンストラクター引数の型ヒントを調べることにより、オブジェクトとそのすべての依存関係を再帰的に自動的に解決する機能の事です。地味な機能ですが、保守性向上の為、コンストラクター・インジェクションは必須と考えて実装することにしました。また、DIコンテナーを使って、サービスプロバイターの仕組みも実装します。この機能については現版のApricotでは使用する場面が無かったので、スタブコントローラの中に例題をつくりました。 尚、DIコンテナーに関する設定ファイル( config/setting/container.setting.php)や初期設定ファイル(config/setup/container.setup.php)は今のところありません。カスタマイズの際は必要に応じて追加して下さい。
 コントローラの Auto Wiring以下は、現状のユーザコントローラのコンストラクタです。 public function __construct() { // モデル $this->user = new User(); ... } このコードを Auto Wiringによって、次のようにすることがここでの目的です。public function __construct(User $user) { // モデル $this->user = $user; ... } Auto Wiringは、コンストラクタの型ヒントによって生成するオブジェクトを判断します。(オブジェクト型でない引数には対応していません)
 ActionInvokerクラスコントローラを生成( new )しているのは、ActionInvoker クラスの invoke() メソッドです。このメソッドを以下のように変更します。 /apricot/core/Foundation - ActionInvoker.php
- <?php namespace Core\Foundation; /** * Request ActionInvoker Class */ class ActionInvoker implements Invoker { .... /** * Invoke action * {@inheritDoc} * @see \Core\Foundation\Invoker::invoke() */ public function invoke() : Response { // Enable auto wiring $container = new \League\Container\Container; $container->delegate(new \League\Container\ReflectionContainer); // Get controller instance $instance = $container->get("\\App\\Controllers\\{$this->controller}"); return call_user_func_array(array($instance, 'invokeAction'), [$this->action, $this->params]); } } 
 - コンテナデリゲートとして ReflectionContainer を登録しAuto Wiringを有効にします。
- コントローラの生成を 単純なnew演算子から Container の get()メソッドに変更します。
 Containerの使い方については、League/Containerを参照して下さい。 
 ユーザコントローラユーザコントローラ (UserController )のコンストラクタを上で述べたように以下のように変更して下さい。 /apricot/app/Controllers - UserController.php
- <?php namespace App\Controllers; use App\Exceptions\ApplicationException; use App\Foundation\Controller; use App\Models\User; use Core\Input; /** * ユーザコントローラ */ class UserController extends Controller { /** * User * @var \App\Models\User */ private $user; /** * ユーザコントローラの生成 */ public function __construct(User $user) { // モデル $this->user = $user; // インターセプター登録 $this->intercept('insert', 'UserInterceptor@insert'); $this->intercept('update', 'UserInterceptor@update'); // トランザクションアクション登録 $this->transactional('insert','update','delete'); } ... } 
 修正が終わったら、Apricotのユーザ一覧画面や編集画面を操作して動作確認をして見て下さい。 
 サービスプロバイターサービスコンテナを使用することで、サービスとサービス間の依存関係を登録しておいて後で取得することができます。例えば、サービスAがモデルBとモデルCを使用しているような場合、サービスコンテナにサービスAを要求すると、自動的にモデルBとCを生成し、それらをサービスAのコンストラクタに与えてサービスAを生成してくれます。 サービスプロバイターは、アプリケーション内の全てのサービスコンテナを登録し整理する方法を提供してくれます。また、サービスプロバイダーではサービスが取得された時点で遅延登録されるため、アプリケーションのパフォーマンス向上にも寄与します。 League/Container でサービスプロバイダーを構築するには以下のステップに従います。 - League/Container の 基本サービスプロバイダー (AbstractServiceProvider) クラスを拡張して 独自のサービスプロバイダーを作ります。
- League/Container のContainerクラスに 独自のサービスプロバイダーを登録します。
 Apricotでは、独自のサービスプロバイダーとして App\Providerクラスを定義し、それを登録するApp\Foundation\Containerクラスをシングルトンとして実装します。サービスの使用者は、App\Foundation\Containerが持っている PSR-11 に準じたget()とhas()を使ってサービスを利用することができます。
 ProviderクラスTODO:/apricot/app - Provider.php
- <?php namespace App; use League\Container\ServiceProvider\AbstractServiceProvider; /** * Provider class for service */ class Provider extends AbstractServiceProvider { /** * The provided array is a way to let the container * know that a service is provided by this service * provider. Every service that is registered via * this service provider must have an alias added * to this array or it will be ignored. * * @var array */ protected $provides = [ // Example 'user', ]; /** * This is where the magic happens, within the method you can * access the container and register or retrieve anything * that you need to, but remember, every alias registered * within this method must be declared in the `$provides` array. */ public function register() { // Example $this->getContainer()->add('user', \App\Models\User::class ); } } 
 
 ContainerクラスTODO:
 TODO: PSR-11についても書く使用法: Container::{メソッド} メソッド 機能 mixed get(string $id) Finds an entry of the container by its identifier and returns it. bool has(string $id) Returns true if the container can return an entry for the given identifier. /apricot/app/Foundation - Container.php
- <?php namespace App\Foundation; use Core\Foundation\Singleton; use App\Provider; /** * Container class for service * * @method static Container getInstance(); * @method static mixed get(string $id) Finds an entry of the container by its identifier and returns it. * @method static bool has(string $id) Returns true if the container can return an entry for the given identifier. */ class Container extends Singleton { /** * Create Container instance. * @return \League\Container\Container */ protected static function createInstance() { $container = new \League\Container\Container; $container->addServiceProvider(new Provider()); return $container; } } 
 テスト実行TODO: StabControllerの例
 
 apricot/ext/di-container.1590475754.txt.gz · 最終更新: 2020/05/26 15:49 by y2sunlight
 
 
 
 





コメント