想理解php依赖注入和控制反转两个概念,就必须搞清楚以下的问题:php
DI——Dependency Injection 依赖注入数组
IoC——Inversion of Control 控制反转框架
##一、参与者都有谁? 函数
答:通常有三方参与者,一个是某个对象;一个是IoC/DI的容器;另外一个是某个对象的外部资源。
IOC/DI容器就是一个全局注册树this
又要名词解释一下,某个对象指的就是任意的、普通的Java对象;code
IoC/DI的容器简单点说就是指用来实现IoC/DI功能的一个框架程序;对象
对象的外部资源指的就是对象须要的,可是是从对象外部获取的,都统称资源,好比:对象须要的其它对象、或者是对象须要的文件资源等等。资源
##二、依赖:谁依赖于谁?为何会有依赖?开发
答:某个对象依赖于IoC/DI的容器。依赖是不可避免的,在一个项目中,各个类之间有各类各样的关系,不可能所有彻底独立,这就造成了依赖。传统的开发是使用其余类时直接调用,这会造成强耦合,这是要避免的。
依赖注入借用容器转移了被依赖对象实现解耦。get
##三、注入:谁注入于谁?到底注入什么?
答:经过容器向对象注入其所须要的外部资源
四、控制反转:谁控制谁?控制什么?为何叫反转?
答:IoC/DI的容器控制对象,主要是控制对象实例的建立。 反转是相对于正向而言的,那么什么算是正向的呢?考虑一下常规状况下的应用程序,若是要在A里面使用C,你会怎么作呢?固然是直接去建立C的对象,也就是说,是在A类中主动去获取所须要的外部资源C,这种状况被称为正向的 。那么什么是反向呢?就是A类再也不主动去获取C,而是被动等待,等待IoC/DI的容器获取一个C的实例,而后反向的注入到A类中。
五、依赖注入和控制反转是同一律念吗?
答:从上面能够看出:依赖注入是从应用程序的角度在描述,能够把依赖注入描述完整点:应用程序依赖容器建立并注入它所须要的外部资源;(程序依赖容器注入外部资源) 而控制反转是从容器的角度在描述,描述完整点:容器控制应用程序,由容器反向的向应用程序注入应用程序所须要的外部资源。 (容器控制应用程序)
下面咱们经过例子来具体看看依赖注入的一些实现方式
#4..简单依赖注入(对象注入)
##4.1 构造器注入
<?php class Book { private $db_conn; public function __construct($db_conn) { $this->db_conn = $db_conn; } }
##4.2 setter注入
<?php class book{ private $db; private $file; function setdb($db){ $this->db=$db; } function setfile($file){ $this->file=$file; } } class file{} class db{} ... class test{ $book = new Book(); $book->setdb(new db()); $book->setfile(new file()); } ?>
##4.3简单依赖注入的缺点(对象注入)
上面俩种方法代码很清晰,可是当咱们须要注入不少个依赖时,意味着又要增长不少行,会比较难以管理
##5.容器注入(高级注入,全局注册树)
比较好的解决办法是 ,*使用注册树模式)
创建一个class做为全部依赖关系的container(容器),在这个class中能够存放、建立、获取、查找须要的依赖关系
<?php class Ioc { protected $db_conn; public static function make_book() { $new_book = new Book(); $new_book->set_db(self::$db_conn); //... //... //其余的依赖注入 return $new_book; } }
此时,若是获取一个book实例,只须要执行$newone = Ioc::makebook();
以上是container的一个具体实例,最好仍是不要把具体的某个依赖注入写成方法,采用registry注册,get获取比较好
?php class Ioc { /** * @var 注册的依赖数组 */ protected static $registry = array(); /** * 添加一个resolve到registry数组中 * @param string $name 依赖标识 * @param object $resolve 一个匿名函数用来建立实例 * @return void */ public static function register($name, Closure $resolve) { static::$registry[$name] = $resolve; } /** * 返回一个实例 * @param string $name 依赖的标识 * @return mixed */ public static function resolve($name) { if ( static::registered($name) ) { $name = static::$registry[$name]; return $name(); } throw new Exception('Nothing registered with that name, fool.'); } /** * 查询某个依赖实例是否存在 * @param string $name id * @return bool */ public static function registered($name) { return array_key_exists($name, static::$registry); } }
如今就能够经过以下方式来注册和注入一个
<?php $book = Ioc::registry('book', function(){ $book = new Book; $book->setdb('...'); $book->setprice('...'); return $book; }); //注入依赖 $book = Ioc::resolve('book'); ?>