依赖注入又叫管制反转,应用过框架的人应该都不生疏。很多人一看名字就感觉是十分高大上的货色,就对它望而生畏,明天解开他它的神秘面纱。废话不多说,间接上代码;
/**
*
* 工具类,应用该类来实现主动依赖注入。*
*/
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,阐明依赖注入胜利。// 打印 $bObj
var_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->cc
params:this is param b
从下面两个例子能够看出咱们创建对象或者调用办法时,基本就不必晓得该类或该办法依赖了那个类。应用反射机制能够轻松的为咱们主动注入所须要的类。