原文地址:https://blog.csdn.net/qq_20678155/article/details/70158374


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

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

        $paramArr = self::getMethodParams($className);
        return (new ReflectionClass($className))->newInstanceArgs($paramArr);
    }

    /**
     * 执行类的方法
     * @param $className
     * @param $methodName
     * @param array $params
     * @return mixed
     * @throws ReflectionException
     */
    public static function make($className, $methodName, $params = [])
    {

        // 获取类的实例
        $instance = self::getInstance($className);
        // 获取该方法所需要依赖注入的参数
        $paramArr = self::getMethodParams($className, $methodName);
        var_dump($paramArr);
        return $instance->{$methodName}(...array_merge($paramArr, $params));
    }

    /**
     * 获得类的方法参数,只获得有类型的参数
     * @param $className
     * @param string $methodsName
     * @return array
     * @throws ReflectionException
     */
    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);
                        var_dump($paramClass->getName());
                        var_dump($args);
                        $paramArr[] = (new ReflectionClass($paramClass->getName()))->newInstanceArgs($args);
                    }
                }
            }
        }

        return $paramArr;
    }
}

class A
{

    protected $cObj;

    /**
     * 用于测试多级依赖注入 B依赖A,A依赖C
     * A constructor.
     * @param C $c
     * @param int $d
     */
    public function __construct(C $c)
    {
        $this->cObj = $c;
    }

    public function aa()
    {

        echo 'this is A->test';
    }

    public function aac()
    {

        echo "<br/>\r\n";
        $this->cObj->cc();
    }
}

class B
{

    protected $aObj;

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

        $this->aObj = $a;
    }

    /**
     * @param C $c     [依赖注入C]
     * @param A $a     [依赖注入A]
     * @param $b       [这个是自己手动填写的参数]
     */
    public function bb(C $c, A $a, $b)
    {

        $c->cc();
        echo "\r\n<br/>";
        $a->aa();
        echo "\r\n<br/>";

        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 , 说明依赖注入成功。

2、测试方法依赖注入

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

从上面两个例子可以看出我们创建对象或者调用方法时,根本就不用知道该类或该方法依赖了那个类。使用反射机制可以轻松的为我们自动注入所需要的类

总结

好了,看到上面的代码是不是觉得很简单,其实只要熟悉php的反射机制,依赖注入并不难实现,上面的代码为了方便理解,所以写的简单除暴,在实际的项目中肯定不会这么简单,比如:会对注入的类和参数进行配置,比如会缓存实例化过的类,下次需要该类的实例时,可以直接使用,而不用在重新初始化,等等。不过相信原理了解了,其他的可以随着项目的需求自己去完善。

Last modification:January 31, 2020
如果觉得我的文章对你有用,请随意赞赏