參考官方簡介的話,PHP 5 具有完整的反射 API,添加了對類、接口、函數、方法和擴展進行反向工程的能力。 此外,反射 API 提供了方法來取出函數、類和方法中的文檔注釋。
YII2框架中示例對于yii2框架,應該都知道di容器,對于di容器的源碼這里也主要講明Container類,先看看平時怎么使用di,就用yii2框架中注釋的示例代碼來展示;
container調用示例namespace app/models;use yii/base/BaseObject;use yii/db/Connection;use yii/di/Container;interface UserFinderInterface function findUser();html' target='_blank'>class UserFinder extends BaseObject implements UserFinderInterface public $db; public function __construct(Connection $db, $config = []) $this- db = $db; parent::__construct($config); public function findUser() class UserLister extends BaseObject public $finder; public function __construct(UserFinderInterface $finder, $config = []) $this- finder = $finder; parent::__construct($config); $container = new Container; $container- set( yii/db/Connection , [ dsn = ... , $container- set( app/models/UserFinderInterface , [ class = app/models/UserFinder , $container- set( userLister , app/models/UserLister $lister = $container- get( userLister // 上述操作相當于下列實現 $db = new /yii/db/Connection([ dsn = ... $finder = new UserFinder($db); $lister = new UserLister($finder);
上面的示例代碼只是實例化了Container類,然后調用set方法注入了其他對象,最后獲取到了依賴與其他對象創建的lister對象,既然只調用了set方法與get方法,那就先從調用最多的set開始看Container代碼。
set方法public function set($class, $definition = [], array $params = []) $this- _definitions[$class] = $this- normalizeDefinition($class, $definition); $this- _params[$class] = $params; unset($this- _singletons[$class]); return $this;}
上面的代碼比較簡潔,調用了類的normalizeDefinition方法,這個一會再說,先說明在該方法中出現的三個屬性的含義
_definitions數組,保存依賴定義
_params數組,保存構造函數的參數
_singletons,保存單例
再看normalizeDefinition方法,該方法主要作用是規范類定義
protected function normalizeDefinition($class, $definition) if (empty($definition)) { // 為空 return [ class = $class]; } elseif (is_string($definition)) { // 為字符串 return [ class = $definition]; } elseif (is_callable($definition, true) || is_object($definition)) { // 檢驗是否為可調用函數或者對象 return $definition; } elseif (is_array($definition)) { // 檢測是否為數組 if (!isset($definition[ class ])) { if (strpos($class, // ) !== false) { $definition[ class ] = $class; } else { throw new InvalidConfigException( A class definition requires a class member. return $definition; throw new InvalidConfigException( Unsupported definition type for / $class/ : . gettype($definition));}
上述代碼中已做了一些判斷注釋,不難發現最后需要返回的definition變量需要為數組格式,或者可調用函數與對象,注意回到剛開始的調用示例代碼,definition變量分別有數組格式不帶class鍵,
數組格式帶class鍵,與字符串類型。到底set方法調用已完畢,從源碼中分析基本上看不到反射的影子,也就是些傳入參數格式兼容處理再寫入類屬性,接著來看下示例代碼中的get方法吧。
public function get($class, $params = [], $config = []) if (isset($this- _singletons[$class])) { // 直接返回單例 return $this- _singletons[$class]; } elseif (!isset($this- _definitions[$class])) { // 調用bulid return $this- build($class, $params, $config); $definition = $this- _definitions[$class]; if (is_callable($definition, true)) { // 可調用函數情況 $params = $this- resolveDependencies($this- mergeParams($class, $params)); $object = call_user_func($definition, $this, $params, $config); } elseif (is_array($definition)) { // 數組 $concrete = $definition[ class unset($definition[ class $config = array_merge($definition, $config); $params = $this- mergeParams($class, $params); if ($concrete === $class) { $object = $this- build($class, $params, $config); } else { $object = $this- get($concrete, $params, $config); } elseif (is_object($definition)) { // 對象直接保存到單例屬性集合中去 return $this- _singletons[$class] = $definition; } else { throw new InvalidConfigException( Unexpected object definition type: . gettype($definition)); if (array_key_exists($class, $this- _singletons)) { // singleton $this- _singletons[$class] = $object; return $object;}
上述代碼,簡要劃分一下,請稍作瀏覽,后面會繼續講述,先說明屬性_definitions集合中不存在的情況,即調用build,這個一會說明,再看如果存在相關class鍵的情況,下面會做幾種情況的處理,
鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。
新聞熱點
疑難解答