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程序设计有所帮助。