PHP基于反射机制实现自动依赖注入的方法详解
本文实例讲述了PHP基于反射机制实现自动依赖注入的方法。分享给大家供大家参考,具体如下:
依赖注入又叫控制反转,使用过框架的人应该都不陌生。很多人一看名字就觉得是非常高大上的东西,就对它望而却步,今天抽空研究了下,解开他它的神秘面纱。废话不多说,直接上代码;
/** * *工具类,使用该类来实现自动依赖注入。 * */ classIoc{ //获得类的对象实例 publicstaticfunctiongetInstance($className){ $paramArr=self::getMethodParams($className); return(newReflectionClass($className))->newInstanceArgs($paramArr); } /** *执行类的方法 *@param[type]$className[类名] *@param[type]$methodName[方法名称] *@param[type]$params[额外的参数] *@return[type][description] */ publicstaticfunctionmake($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] */ protectedstaticfunctiongetMethodParams($className,$methodsName='__construct'){ //通过反射获得该类 $class=newReflectionClass($className); $paramArr=[];//记录参数,和参数类型 //判断该类是否有构造函数 if($class->hasMethod($methodsName)){ //获得构造函数 $construct=$class->getMethod($methodsName); //判断构造函数是否有参数 $params=$construct->getParameters(); if(count($params)>0){ //判断参数类型 foreach($paramsas$key=>$param){ if($paramClass=$param->getClass()){ //获得参数类型名称 $paramClassName=$paramClass->getName(); //获得参数类型 $args=self::getMethodParams($paramClassName); $paramArr[]=(newReflectionClass($paramClass->getName()))->newInstanceArgs($args); } } } } return$paramArr; } }
上面的代码使用php的反射函数,创建了一个容器类,使用该类来实现其他类的依赖注入功能。上面的依赖注入分为两种,一种是构造函数的依赖注入,一种是方法的依赖注入。我们使用下面三个类来做下测试。
classA{ protected$cObj; /** *用于测试多级依赖注入B依赖A,A依赖C *@paramC$c[description] */ publicfunction__construct(C$c){ $this->cObj=$c; } publicfunctionaa(){ echo'thisisA->test'; } publicfunctionaac(){ $this->cObj->cc(); } } classB{ protected$aObj; /** *测试构造函数依赖注入 *@paramA$a[使用引来注入A] */ publicfunction__construct(A$a){ $this->aObj=$a; } /** *[测试方法调用依赖注入] *@paramC$c[依赖注入C] *@paramstring$b[这个是自己手动填写的参数] *@return[type][description] */ publicfunctionbb(C$c,$b){ $c->cc(); echo"\r\n"; echo'params:'.$b; } /** *验证依赖注入是否成功 *@return[type][description] */ publicfunctionbbb(){ $this->aObj->aac(); } } classC{ publicfunctioncc(){ echo'thisisC->cc'; } }
测试构造函数的依赖注入
//使用Ioc来创建B类的实例,B的构造函数依赖A类,A的构造函数依赖C类。 $bObj=Ioc::getInstance('B'); $bObj->bbb();//输出:thisisC->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',['thisisparamb']); //输出结果,可以看出依赖注入成功。 thisisC->cc params:thisisparamb
从上面两个例子可以看出我们创建对象或者调用方法时,根本就不用知道该类或该方法依赖了那个类。使用反射机制可以轻松的为我们自动注入所需要的类。
总结
好了,看到上面的代码是不是觉得很简单,其实只要熟悉php的反射机制,依赖注入并不难实现,上面的代码为了方便理解,所以写的简单除暴,在实际的项目中肯定不会这么简单,比如:会对注入的类和参数进行配置,比如会缓存实例化过的类,下次需要该类的实例时,可以直接使用,而不用在重新初始化,等等。不过相信原理了解了,其他的可以随着项目的需求自己去完善。
更多关于PHP相关内容感兴趣的读者可查看本站专题:《php面向对象程序设计入门教程》、《PHP基本语法入门教程》、《PHP运算与运算符用法总结》、《PHP网络编程技巧总结》、《PHP数组(Array)操作技巧大全》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总》
希望本文所述对大家PHP程序设计有所帮助。