DI依赖注入,IOC控制反转。
Demo:
/**
*
* 工具类,使用该类来实现自动依赖注入。
*
*/
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的反射机制创建一个容器类,上述的DI分为构造函数注入及属性注入,测试代码如下:
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
上述简单demo可以看出,在创建对象或调用方法时,根本就不用知道该类或该方法依赖了哪个类,使用反射机制可以轻松的为我们自动注入所需要的类。