理解什麼是Di/IoC,依賴注入/控制反轉。兩者說的是一個東西,是當下流行的一種設計模式。大致的意思就是,準備一個盒子(容器),事先將項目中可能用到的類扔進去,在項目中直接從容器中拿,也就是避免了直接在項目中到處new,造成大量耦合。取而代之的是在項目類裡面增設 setDi()和getDi()方法,通過Di同一管理類。
直接上代碼吧.
Di容器類:
class Di implements \ArrayAccess{ private $_bindings = array();//服務列表 private $_instances = array();//已經執行個體化的服務 //擷取服務 public function get($name,$params=array()){ //先從已經執行個體化的列表中尋找 if(isset($this->_instances[$name])){ return $this->_instances[$name]; } //檢測有沒有註冊該服務 if(!isset($this->_bindings[$name])){ return null; } $concrete = $this->_bindings[$name]['class'];//對象具體註冊內容 $obj = null; //匿名函數方式 if($concrete instanceof \Closure){ $obj = call_user_func_array($concrete,$params); }elseif(is_string($concrete)){//字串方式 if(empty($params)){ $obj = new $concrete; }else{ //帶參數的類執行個體化,使用反射 $class = new \ReflectionClass($concrete); $obj = $class->newInstanceArgs($params); } } //如果是共用服務,則寫入_instances列表,下次直接取回 if($this->_bindings[$name]['shared'] == true && $obj){ $this->_instances[$name] = $obj; } return $obj; } //檢測是否已經綁定 public function has($name){ return isset($this->_bindings[$name]) or isset($this->_instances[$name]); } //卸載服務 public function remove($name){ unset($this->_bindings[$name],$this->_instances[$name]); } //設定服務 public function set($name,$class){ $this->_registerService($name, $class); } //設定共用服務 public function setShared($name,$class){ $this->_registerService($name, $class, true); } //註冊服務 private function _registerService($name,$class,$shared=false){ $this->remove($name); if(!($class instanceof \Closure) && is_object($class)){ $this->_instances[$name] = $class; }else{ $this->_bindings[$name] = array("class"=>$class,"shared"=>$shared); } } //ArrayAccess介面,檢測服務是否存在 public function offsetExists($offset) { return $this->has($offset); } //ArrayAccess介面,以$di[$name]方式擷取服務 public function offsetGet($offset) { return $this->get($offset); } //ArrayAccess介面,以$di[$name]=$value方式註冊服務,非共用 public function offsetSet($offset, $value) { return $this->set($offset,$value); } //ArrayAccess介面,以unset($di[$name])方式卸載服務 public function offsetUnset($offset) { return $this->remove($offset); } }
<?php header("Content-Type:text/html;charset=utf8"); class A{ public $name; public $age; public function __construct($name=""){ $this->name = $name; } } include "Di.class.php"; $di = new Di(); //匿名函數方式註冊一個名為a1的服務 $di->setShared('a1',function($name=""){ return new A($name); }); //直接以類名方式註冊 $di->set('a2','A'); //直接傳入執行個體化的對象 $di->set('a3',new A("小唐")); $a1 = $di->get('a1',array("小李")); echo $a1->name."<br/>";//小李 $a1_1 = $di->get('a1',array("小王")); echo $a1->name."<br/>";//小李 echo $a1_1->name."<br/>";//小李 $a2 = $di->get('a2',array("小張")); echo $a2->name."<br/>";//小張 $a2_1 = $di->get('a2',array("小徐")); echo $a2->name."<br/>";//小張 echo $a2_1->name."<br/>";//小徐 $a3 = $di['a3'];//可以直接通過數組方式擷取服務物件 echo $a3->name."<br/>";//小唐
通過set和setShared方式註冊服務,支援 匿名函數,類名字串,已經執行個體化的對象.
兩者的區別是:set方式註冊的,每次擷取的時候都會重新執行個體化setShared方式的,則只執行個體化一次,也就是所謂的單例模式
當然,對於直接註冊已經執行個體化的對象,如上代碼中的a3服務,set和setShared效果是一樣的。
通過$di->get()擷取服務,可接受兩個參數,第一個參數是服務名,比如a1,a2,a3是必須的,第二個參數是一個數組,第二個參數會被當作匿名函數中的參數或者類建構函式裡的參數傳進去,參考call_user_func_array()。
刪除服務則可以通過
unset($di['a1']);or$di->remove('a1');判斷是否包含一個服務可以通過isset($di['a1']);or$di->has('a1');就這麼多了。