PHP基于反射机制实现自动依赖注入的方法详解
程序员文章站
2022-07-05 12:10:41
本文实例讲述了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
从上面两个例子可以看出我们创建对象或者调用方法时,根本就不用知道该类或该方法依赖了那个类。使用反射机制可以轻松的为我们自动注入所需要的类。
总结
好了,看到上面的代码是不是觉得很简单,其实只要熟悉php的反射机制,依赖注入并不难实现,上面的代码为了方便理解,所以写的简单除暴,在实际的项目中肯定不会这么简单,比如:会对注入的类和参数进行配置,比如会缓存实例化过的类,下次需要该类的实例时,可以直接使用,而不用在重新初始化,等等。不过相信原理了解了,其他的可以随着项目的需求自己去完善。
更多关于php相关内容感兴趣的读者可查看本站专题:《php面向对象程序设计入门教程》、《php基本语法入门教程》、《php运算与运算符用法总结》、《php网络编程技巧总结》、《php数组(array)操作技巧大全》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总》
希望本文所述对大家php程序设计有所帮助。