依赖注入又叫管制反转,应用过框架的人应该都不生疏。很多人一看名字就感觉是十分高大上的货色,就对它望而生畏,明天解开他它的神秘面纱。废话不多说,间接上代码;

/**** 工具类,应用该类来实现主动依赖注入。**/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的反射函数,创立了一个容器类,应用该类来实现其余类的依赖注入性能。下面的依赖注入分为两种,一种是构造函数的依赖注入,一种是办法的依赖注入。 咱们应用上面三个类来做下测试。

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';    }}

测试构造函数的依赖注入:

// 应用Ioc来创立B类的实例,B的构造函数依赖A类,A的构造函数依赖C类。$bObj = Ioc::getInstance('B');$bObj->bbb(); // 输入:this is C->cc , 阐明依赖注入胜利。// 打印$bObjvar_dump($bObj);// 打印后果,能够看出B中有A实例,A中有C实例,阐明依赖注入胜利。object(B)#3 (1) {  ["aObj":protected]=>  object(A)#7 (1) {    ["cObj":protected]=>    object(C)#10 (0) {    }  }}

测试方法依赖注入:

Ioc::make('B', 'bb', ['this is param b']);// 输入后果,能够看出依赖注入胜利。this is C->ccparams:this is param b

 从下面两个例子能够看出咱们创建对象或者调用办法时,基本就不必晓得该类或该办法依赖了那个类。应用反射机制能够轻松的为咱们主动注入所须要的类。