成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

PHP如何實現依賴注入

開發 后端
控制反轉(Inversion of Control,英文縮寫為IoC)是框架的重要特征。控制反轉(IOC)是一種思想,依賴注入(DI)是實施這種思想的方法。

[[200267]]

摘要: 控制反轉(Inversion of Control,英文縮寫為IoC)是框架的重要特征??刂品崔D(IOC)是一種思想,依賴注入(DI)是實施這種思想的方法。

高層模塊不應該依賴于底層模塊,兩個都應該依賴抽象。

抽象不應該依賴于細節,細節應該依賴于抽象。

首先,我們來看一段代碼:

  1. class A{ 
  2.         public function echo() 
  3.         { 
  4.                 echo 'A'.PHP_EOL; 
  5.         } 
  6. class EchoT { 
  7.         protected  $t; 
  8.         public function __construct() 
  9.         { 
  10.               $this->t = new A(); 
  11.         } 
  12.         public function echo(){ 
  13.                 $this->t->echo(); 
  14.         } 
  15.  

初始,我們都使用new 的方式在內部進行,EchoT類嚴重依賴于類A。每當類A變化時,EchoT類也得進行變化。

我們優化一下代碼

  1. class EchoT { 
  2.         protected  $t; 
  3.         public function __construct($t)  //構造器注入由構造器注入到其中 
  4.         { 
  5.               $this->t = $t; 
  6.         }  

可以看到,這樣做的話。很大程序上,我們對程序進行了解耦。類A無論你如何變動,EchoT類是不需要變動的。不再依賴于A。但是新問題又來了,我們現在只有A,萬一來了B,來了CDEFG怎么辦。

面向接口 

  1. interface T{ 
  2.         public function echo(); 
  3.  
  4. class A{ 
  5.         public function echo() 
  6.         { 
  7.                 echo 'A'.PHP_EOL; 
  8.         } 
  9.  
  10. class B implements T{ 
  11.         public function echo() 
  12.         { 
  13.                 echo 'B'.PHP_EOL; 
  14.         } 
  15. class EchoT { 
  16.         protected  $t; 
  17.         public function __construct(T $t)  //構造器注入由構造器注入到其中 
  18.         { 
  19.               $this->t = $t; 
  20.         } 
  21.         public function echo(){ 
  22.                 $this->t->echo(); 
  23.         } 
  24.  

將T抽象出為接口,這樣,EchoT類中的echo方法變成一個抽象的方法,不到運行那一刻,不知道他們的Method方式是怎么實現的。

工廠

  1. function getT($str) { 
  2.     if(class_exists($str)){ 
  3.         return new $str(); 
  4.         } 
  5.  

T要使用哪個是不明確的,因此,我們可以將其工廠化?!究瓷先ズ芎唵?,在DI實際上有體現】

DI(重點來了)

首先,我們看一下PHP的psr規范。

http://www.php-fig.org/psr/psr-11/

官方定義的接口

Psr\Container\ContainerInterface

包含兩個方法

function get($id);

function has($id);

仔細看上面的工廠,是不是和get($id)很一致,PHP官方將其定義為容器(Container,我個人理解,就是一個復雜的工廠)

dependency injection container

依賴注入容器

  1. namespace Core; 
  2. use Psr\Container\ContainerInterface; 
  3. class Container implements ContainerInterface 
  4.         protected $instance = [];//對象存儲的數組 
  5.         public function __construct($path) { 
  6.                 $this->_autoload($path);  //首先我們要自動加載  psr-autoload 
  7.         } 
  8.  
  9.         public function build($className) 
  10.         { 
  11.                 if(is_string($className) and $this->has($className)) { 
  12.                         return $this->get($className); 
  13.                 } 
  14.                 //反射 
  15.                 $reflector = new \ReflectionClass($className); 
  16.                 if (!$reflector->isInstantiable()) { 
  17.                         throw new \Exception("Can't instantiate ".$className); 
  18.                 } 
  19.                 // 檢查類是否可實例化, 排除抽象類abstract和對象接口interface 
  20.                 if (!$reflector->isInstantiable()) { 
  21.                         throw new \Exception("Can't instantiate ".$className); 
  22.                 } 
  23.                 /** @var \ReflectionMethod $constructor 獲取類的構造函數 */ 
  24.                 $constructor = $reflector->getConstructor(); 
  25.                 // 若無構造函數,直接實例化并返回 
  26.                 if (is_null($constructor)) { 
  27.                         return new $className; 
  28.                 } 
  29.                 // 取構造函數參數,通過 ReflectionParameter 數組返回參數列表 
  30.                 $parameters = $constructor->getParameters(); 
  31.                 // 遞歸解析構造函數的參數 
  32.                 $dependencies = $this->getDependencies($parameters); 
  33.                 // 創建一個類的新實例,給出的參數將傳遞到類的構造函數。 
  34.                 $class =  $reflector->newInstanceArgs($dependencies); 
  35.                 $this->instance[$className] = $class; 
  36.                 return $class; 
  37.         } 
  38.  
  39.         /** 
  40.          * @param array $parameters 
  41.          * @return array 
  42.          */ 
  43.         public function getDependencies(array $parameters) 
  44.         { 
  45.                 $dependencies = []; 
  46.                 /** @var \ReflectionParameter $parameter */ 
  47.                 foreach ($parameters as $parameter) { 
  48.                         /** @var \ReflectionClass $dependency */ 
  49.                         $dependency = $parameter->getClass(); 
  50.                         if (is_null($dependency)) { 
  51.                                 // 是變量,有默認值則設置默認值 
  52.                                 $dependencies[] = $this->resolveNonClass($parameter); 
  53.                         } else { 
  54.                                 // 是一個類,遞歸解析 
  55.                                 $dependencies[] = $this->build($dependency->name); 
  56.                         } 
  57.                 } 
  58.                 return $dependencies; 
  59.         } 
  60.  
  61.         /** 
  62.          * @param \ReflectionParameter $parameter 
  63.          * @return mixed 
  64.          * @throws \Exception 
  65.          */ 
  66.         public function resolveNonClass(\ReflectionParameter $parameter) 
  67.         { 
  68.                 // 有默認值則返回默認值 
  69.                 if ($parameter->isDefaultValueAvailable()) { 
  70.                         return $parameter->getDefaultValue(); 
  71.                 } 
  72.                 throw new \Exception($parameter->getName().' must be not null'); 
  73.         } 
  74.         /** 
  75.          * 參照psr-autoload規范 
  76.          * @param $path 
  77.          */ 
  78.         public function _autoload($path) { 
  79.                 spl_autoload_register(function(string $class) use ($path) { 
  80.                         $file = DIRECTORY_SEPARATOR.str_replace('\\',DIRECTORY_SEPARATOR, $class).'.php'; 
  81.                         if(is_file($path.$file)) { 
  82.                                 include($path.$file); 
  83.                                 return true
  84.                         } 
  85.                         return false
  86.                 }); 
  87.         } 
  88.  
  89.         public function get($id) 
  90.         { 
  91.                 if($this->has($id)) { 
  92.                         return $this->instance[$id]; 
  93.                 } 
  94.                 if(class_exists($id)){ 
  95.                         return $this->build($id); 
  96.                 } 
  97.                 throw new ClassNotFoundException('class not found');  //實現的PSR規范的異常 
  98.         } 
  99.  
  100.         public function has($id) 
  101.         { 
  102.                 return isset($this->instance[$id]) ? true : false
  103.         } 
  104.  

使用示例

  1. $container = new Container('../');//假設這是路徑 
  2. $echoT = $container->get(\Test\EchoT::class);     //假設echoT類的命名空間是\Test 
  3. $echoT->echo();  

這個時候,會出現一個問題:

  1. // 檢查類是否可實例化, 排除抽象類abstract和對象接口interface 
  2.                 if (!$reflector->isInstantiable()) { 
  3.                         throw new \Exception("Can't instantiate ".$className);  

因為接口T是無法實例化的,所以,一般在程序內,我們都加上別名(參照laravel框架)

  1. $container->alisa(\Test\T::class,\Test\T\A::class); //指定接口T使用類A(控制反轉) 

針對接口

下面是alias方法 

  1. public function alias(string $key, $class, bool $singleton = true)  
  2.        { 
  3.                if($singleton) { 
  4.                        $this->singleton[] = $class; 
  5.                } 
  6.                $this->aliases[$key] = $class; 
  7.                return $this; 
  8.        } 
  9.    //同時,我們需要在build的時候進行判斷是否為別名 
  10. public function build($className) 
  11.        { 
  12.                if(is_string($className) and $this->has($className)) { 
  13.                        return $this->get($className); 
  14.                } 
  15.                if(isset($this->aliases[$className])) { 
  16.                        if(is_object($this->aliases[$className])) { 
  17.                               return $this->aliases[$className]; 
  18.                        } 
  19.                        $className = $this->aliases[$className]; 
  20.                }  

就此,一個簡單的PHP容器就實現了。 

責任編輯:龐桂玉 來源: o0無憂亦無怖的博客
相關推薦

2018-03-12 10:02:30

PHP依賴注入

2015-09-02 11:22:36

JavaScript實現思路

2011-05-31 10:00:21

Android Spring 依賴注入

2010-05-19 14:54:33

2020-08-06 00:14:16

Spring IoC依賴注入開發

2023-07-11 09:14:12

Beanquarkus

2022-12-29 08:54:53

依賴注入JavaScript

2011-03-01 13:45:41

Spring3Annotation

2016-10-20 19:36:01

androiddagger2依賴注入

2025-02-17 00:00:55

NET開發依賴注入

2013-12-09 09:57:37

2022-04-30 08:50:11

控制反轉Spring依賴注入

2019-09-18 18:12:57

前端javascriptvue.js

2013-12-13 10:45:26

2012-12-19 10:36:06

2010-10-14 09:05:36

ASP.NET MVC

2024-12-30 12:00:00

.NET Core依賴注入屬性注入

2024-04-01 00:02:56

Go語言代碼

2025-01-13 00:13:59

VSCode架構依賴注入

2024-05-27 00:13:27

Go語言框架
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产婷婷精品av在线 | 中文字幕日韩一区 | 欧美日韩国产精品一区 | 日韩精品一区二区三区中文在线 | 中文字幕一区二区在线观看 | 欧美在线综合 | 免费看a | 久久久蜜臀国产一区二区 | 亚洲+变态+欧美+另类+精品 | 亚洲精品免费在线 | 99久久电影 | 成人午夜精品 | av黄在线观看 | 国产欧美日韩一区 | 999久久久久久久 | 久久99视频这里只有精品 | 亚洲精品日韩在线观看 | www.夜夜草 | 黑人巨大精品 | 一久久久 | 拍真实国产伦偷精品 | 黄色网页在线观看 | 国产精品欧美一区二区三区 | 国产精品成人一区 | 欧美视频在线看 | 成人精品区 | 久久综合激情 | 伊人网一区| 日韩精品一区二区三区在线观看 | 日韩在线h | 中国一级特黄毛片大片 | 色888www视频在线观看 | 国产精品久久久久久久久婷婷 | 成人欧美一区二区三区 | 亚洲精品一区二区三区 | 日本五月婷婷 | 毛片网站在线观看视频 | а_天堂中文最新版地址 | 91中文字幕在线 | 日韩一区二区三区av | 色狠狠一区 |