Di/IoC, 즉 종속성 주입/제어 반전이 무엇인지 이해하세요. 두 사람은 같은 것을 이야기하고 있는데, 이는 현재 인기 있는 디자인 패턴이다. 일반적인 의미는 상자(컨테이너)를 준비하고, 프로젝트에서 사용할 수 있는 클래스를 미리 그 안에 던지고, 프로젝트의 컨테이너에서 직접 가져오는 것입니다. 커플링이 많음. 대신 프로젝트 클래스에 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'); 就这么多了。
를 통해 서비스를 삭제할 수 있습니다. 관련 권장 사항:
위 내용은 PHP로 경량 컨테이너를 작성하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!