本文实例讲述了PHP基于反射机制实现自动依赖注入的方法。分享给大家供大家参考,具体如下:
依赖注入又叫控制反转,使用过框架的人应该都不陌生。很多人一看名字就觉得是非常高大上的东西,就对它望而却步,今天抽空研究了下,解开他它的神秘面纱。废话不多说,直接上代码;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
|
/**
*
* 工具类,使用该类来实现自动依赖注入。
*
*/
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的反射函数,创建了一个容器类,使用该类来实现其他类的依赖注入功能。上面的依赖注入分为两种,一种是构造函数的依赖注入,一种是方法的依赖注入。 我们使用下面三个类来做下测试。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
|
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
2
3
4
5
6
7
8
9
10
11
12
13
14
|
// 使用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) {
}
}
}
|
测试方法依赖注入
1
2
3
4
|
Ioc::make( 'B' , 'bb' , [ 'this is param b' ]);
// 输出结果,可以看出依赖注入成功。
this is C->cc
params:this is param b
|
从上面两个例子可以看出我们创建对象或者调用方法时,根本就不用知道该类或该方法依赖了那个类。使用反射机制可以轻松的为我们自动注入所需要的类。
总结
好了,看到上面的代码是不是觉得很简单,其实只要熟悉php的反射机制,依赖注入并不难实现,上面的代码为了方便理解,所以写的简单除暴,在实际的项目中肯定不会这么简单,比如:会对注入的类和参数进行配置,比如会缓存实例化过的类,下次需要该类的实例时,可以直接使用,而不用在重新初始化,等等。不过相信原理了解了,其他的可以随着项目的需求自己去完善。
希望本文所述对大家PHP程序设计有所帮助。
原文链接:http://blog.csdn.net/qq_20678155/article/details/70158374