ホームページ  >  記事  >  バックエンド開発  >  PHPのリフレクション機構を利用した自動依存性注入の詳細説明

PHPのリフレクション機構を利用した自動依存性注入の詳細説明

韦小宝
韦小宝オリジナル
2018-03-14 12:47:161765ブラウズ

この記事では、PHP がリフレクション メカニズムと自動的な 依存関係注入 を実装する方法について説明します。PHP リフレクション メカニズムと自動依存関係注入についてよく知らない学生は、この記事を一緒に見てみましょう。

依存性注入は制御の反転とも呼ばれ、フレームワークを使用したことがある人なら誰でもよく知っているはずです。名前を見るととても高級なものだと思って敬遠してしまう人も多いので、今回は時間をかけて調べてその謎を解き明かしてみました。早速、コードの説明に入りましょう

/**
*
* 工具类,使用该类来实现自动依赖注入。
*
*/
class Ioc {

    // 获得类的对象实例
    public static function getInstance($className) {

        $paramArr = self::getMethodParams($className);

        return (new ReflectionClass($className))->newInstanceArgs($paramArr);
    }

    /**
     * 执行类的方法
     * @param  [type] $className  [类名]
     * @param  [type] $methodName [方法名称]
     * @param  [type] $params     [额外的参数]
     * @return [type]             [description]
     */
    public static function make($className, $methodName, $params = []) {

        // 获取类的实例
        $instance = self::getInstance($className);

        // 获取该方法所需要依赖注入的参数
        $paramArr = self::getMethodParams($className, $methodName);

        return $instance->{$methodName}(...array_merge($paramArr, $params));
    }

    /**
     * 获得类的方法参数,只获得有类型的参数
     * @param  [type] $className   [description]
     * @param  [type] $methodsName [description]
     * @return [type]              [description]
     */
    protected static function getMethodParams($className, $methodsName = 'construct') {

        // 通过反射获得该类
        $class = new ReflectionClass($className);
        $paramArr = []; // 记录参数,和参数类型

        // 判断该类是否有构造函数
        if ($class->hasMethod($methodsName)) {
            // 获得构造函数
            $construct = $class->getMethod($methodsName);

            // 判断构造函数是否有参数
            $params = $construct->getParameters();

            if (count($params) > 0) {

                // 判断参数类型
                foreach ($params as $key => $param) {

                    if ($paramClass = $param->getClass()) {

                        // 获得参数类型名称
                        $paramClassName = $paramClass->getName();

                        // 获得参数类型
                        $args = self::getMethodParams($paramClassName);
                        $paramArr[] = (new ReflectionClass($paramClass->getName()))->newInstanceArgs($args);
                    }
                }
            }
        }

        return $paramArr;
    }
}

上記のコードは、PHP のリフレクション関数を使用してコンテナ クラスを作成し、このクラスを使用して他のクラスの依存関係注入関数を実装します。上記の依存性注入は、コンストラクタの依存性注入とメソッドの依存性注入の2種類に分けられます。 テストには次の 3 つのクラスを使用します。

class A {    protected $cObj;    /**
     * 用于测试多级依赖注入 B依赖A,A依赖C
     * @param C $c [description]
     */
    public function construct(C $c) {

        $this->cObj = $c;
    }    public function aa() {

        echo 'this is A->test';
    }    public function aac() {

        $this->cObj->cc();
    }
}

class B {    protected $aObj;    /**
     * 测试构造函数依赖注入
     * @param A $a [使用引来注入A]
     */
    public function construct(A $a) {

        $this->aObj = $a;
    }    /**
     * [测试方法调用依赖注入]
     * @param  C      $c [依赖注入C]
     * @param  string $b [这个是自己手动填写的参数]
     * @return [type]    [description]
     */
    public function bb(C $c, $b) {

        $c->cc();
        echo "\r\n";

        echo 'params:' . $b;
    }    /**
     * 验证依赖注入是否成功
     * @return [type] [description]
     */
    public function bbb() {

        $this->aObj->aac();
    }
}

class C {    public function cc() {

        echo 'this is C->cc';
    }
}

1. テスト コンストラクターの依存関係の注入

// 使用Ioc来创建B类的实例,B的构造函数依赖A类,A的构造函数依赖C类。
$bObj = Ioc::getInstance('B');
$bObj->bbb(); // 输出:this is C->cc , 说明依赖注入成功。

// 打印$bObj
var_dump($bObj);

// 打印结果,可以看出B中有A实例,A中有C实例,说明依赖注入成功。
object(B)#3 (1) {
  ["aObj":protected]=>
  object(A)#7 (1) {
    ["cObj":protected]=>
    object(C)#10 (0) {
    }
  }
}

2. テスト メソッドの依存関係の注入

Ioc::make('B', 'bb', ['this is param b']);

// 输出结果,可以看出依赖注入成功。
this is C->cc
params:this is param b

上記の 2 つの例から、オブジェクトを作成する またはメソッドを呼び出すときに、クラスやメソッドを知ることはそのクラスに依存します。リフレクション メカニズムを使用すると、必要なクラスを簡単に自動的に挿入できます。

概要

さて、上記のコードは非常に単純だと思いますか? 実際、PHP のリフレクション メカニズムに精通している限り、上記のコードは実装するのが難しくありません。実際のプロジェクトでは、違反の解除は決して簡単ではありません。たとえば、インスタンス化されたクラスは次回にキャッシュされます。クラスが必要な場合は、再初期化中などの代わりに直接使用できます。ただし、原則を理解すれば、残りはプロジェクトのニーズに応じて改善できると思います。

以上がPHPのリフレクション機構を利用した自動依存性注入の詳細説明の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。