欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP基于反射机制实现自动依赖注入的详细教程

程序员文章站 2022-04-15 12:51:45
本文实例讲述了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的反射函数,创建了一个容器类,使用该类来实现其他类的依赖注入功能。上面的依赖注入分为两种,一种是构造函数的依赖注入,一种是方法的依赖注入。 我们使用下面三个类来做下测试。

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

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