摘要:出于需要,我們自己造一個輕量級的輪子,為了保持規(guī)范,我們基于來實現(xiàn)。這樣我們就擁有了一個功能豐富,使用方便的輕量級容器了,趕快整合到你的項目中去吧。
什么是容器
在開發(fā)過程中,經(jīng)常會用到的一個概念就是依賴注入。我們借助依懶注入來解耦代碼,選擇性的按需加載服務(wù),而這些通常都是借助容器來實現(xiàn)。
容器實現(xiàn)對對象的統(tǒng)一管理,并且確保對象實例的唯一性
容器可以很輕易的找到有很多實現(xiàn)示例,如 PHP-DI 、 YII-DI 等各種實現(xiàn),通常他們要么大而全,要么高度適配特定業(yè)務(wù),與實際需要存在沖突。
出于需要,我們自己造一個輕量級的輪子,為了保持規(guī)范,我們基于 PSR-11 來實現(xiàn)。
PSR-11
PSR 是 php-fig 提供的標(biāo)準(zhǔn)化建議,雖然不是官方組織,但是得到廣泛認(rèn)可。PSR-11 提供了容器接口。它包含 ContainerInterface 和 兩個異常接口,并提供使用建議。
/** * Describes the interface of a container that exposes methods to read its entries. */ interface ContainerInterface { /** * Finds an entry of the container by its identifier and returns it. * * @param string $id Identifier of the entry to look for. * * @throws NotFoundExceptionInterface No entry was found for **this** identifier. * @throws ContainerExceptionInterface Error while retrieving the entry. * * @return mixed Entry. */ public function get($id); /** * Returns true if the container can return an entry for the given identifier. * Returns false otherwise. * * `has($id)` returning true does not mean that `get($id)` will not throw an exception. * It does however mean that `get($id)` will not throw a `NotFoundExceptionInterface`. * * @param string $id Identifier of the entry to look for. * * @return bool */ public function has($id); }
實現(xiàn)示例
我們先來實現(xiàn)接口中要求的兩個方法
abstract class AbstractContainer implements ContainerInterface { protected $resolvedEntries = []; /** * @var array */ protected $definitions = []; public function __construct($definitions = []) { foreach ($definitions as $id => $definition) { $this->injection($id, $definition); } } public function get($id) { if (!$this->has($id)) { throw new NotFoundException("No entry or class found for {$id}"); } $instance = $this->make($id); return $instance; } public function has($id) { return isset($this->definitions[$id]); }
實際我們?nèi)萜髦凶⑷氲膶ο笫嵌喾N多樣的,所以我們多帶帶抽出實例化方法。
protected function make($name) { if (isset($this->resolvedEntries[$name])) { return $this->resolvedEntries[$name]; } $definition = $this->definitions[$name]; $params = []; if (is_array($definition) && isset($definition["class"])) { $params = $definition; $definition = $definition["class"]; unset($params["class"]); } $object = $this->reflector($definition, $params); return $this->resolvedEntries[$name] = $object; } public function reflector($concrete, array $params = []) { if ($concrete instanceof Closure) { return $concrete($params); } elseif (is_string($concrete)) { $reflection = new ReflectionClass($concrete); $dependencies = $this->getDependencies($reflection); foreach ($params as $index => $value) { $dependencies[$index] = $value; } return $reflection->newInstanceArgs($dependencies); } elseif (is_object($concrete)) { return $concrete; } } /** * @param ReflectionClass $reflection * @return array */ private function getDependencies($reflection) { $dependencies = []; $constructor = $reflection->getConstructor(); if ($constructor !== null) { $parameters = $constructor->getParameters(); $dependencies = $this->getParametersByDependencies($parameters); } return $dependencies; } /** * * 獲取構(gòu)造類相關(guān)參數(shù)的依賴 * @param array $dependencies * @return array $parameters * */ private function getParametersByDependencies(array $dependencies) { $parameters = []; foreach ($dependencies as $param) { if ($param->getClass()) { $paramName = $param->getClass()->name; $paramObject = $this->reflector($paramName); $parameters[] = $paramObject; } elseif ($param->isArray()) { if ($param->isDefaultValueAvailable()) { $parameters[] = $param->getDefaultValue(); } else { $parameters[] = []; } } elseif ($param->isCallable()) { if ($param->isDefaultValueAvailable()) { $parameters[] = $param->getDefaultValue(); } else { $parameters[] = function ($arg) { }; } } else { if ($param->isDefaultValueAvailable()) { $parameters[] = $param->getDefaultValue(); } else { if ($param->allowsNull()) { $parameters[] = null; } else { $parameters[] = false; } } } } return $parameters; }
如你所見,到目前為止我們只實現(xiàn)了從容器中取出實例,從哪里去提供實例定義呢,所以我們還需要提供一個方法.
/** * @param string $id * @param string | array | callable $concrete * @throws ContainerException */ public function injection($id, $concrete) { if (!is_string($id)) { throw new InvalidArgumentException(sprintf( "The id parameter must be of type string, %s given", is_object($id) ? get_class($id) : gettype($id) )); } if (is_array($concrete) && !isset($concrete["class"])) { throw new ContainerException("數(shù)組必須包含類定義"); } $this->definitions[$id] = $concrete; }
只有這樣嗎?對的,有了這些操作我們已經(jīng)有一個完整的容器了,插箱即用。
不過為了使用方便,我們可以再提供一些便捷的方法,比如數(shù)組式訪問。
class Container extends AbstractContainer implements ArrayAccess { public function offsetExists($offset) { return $this->has($offset); } public function offsetGet($offset) { return $this->get($offset); } public function offsetSet($offset, $value) { return $this->injection($offset, $value); } public function offsetUnset($offset) { unset($this->resolvedEntries[$offset]); unset($this->definitions[$offset]); } }
這樣我們就擁有了一個功能豐富,使用方便的輕量級容器了,趕快整合到你的項目中去吧。
點擊這里查看完整代碼
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/30042.html
摘要:是一個基于擴展實現(xiàn)的輕量級高性能的常駐內(nèi)存型的和應(yīng)用服務(wù)框架高度封裝了,,服務(wù)器,以及基于實現(xiàn)可擴展的服務(wù),同時支持包方式安裝部署項目。基于實用,抽象事件處理類,實現(xiàn)與底層的回調(diào)的解耦,支持同步異步調(diào)用,內(nèi)置等常用組件等。 swoolefy swoolefy是一個基于swoole擴展實現(xiàn)的輕量級高性能的常駐內(nèi)存型的API和Web應(yīng)用服務(wù)框架,高度封裝了http,websocket,ud...
摘要:做了一次分享,主題使用搭建開發(fā)環(huán)境,簡單介紹了一下的概念,演示了使用構(gòu)建全套環(huán)境。應(yīng)場景通常于如下場景應(yīng)的動化打包和發(fā)布動化測試和持續(xù)集成發(fā)布在服務(wù)型環(huán)境中部署和調(diào)整數(shù)據(jù)庫或其他的后臺應(yīng)從頭編譯或者擴展現(xiàn)有的或平臺來搭建的環(huán)境。 做了一次分享,主題《使用 Docker 搭建開發(fā)環(huán)境》,簡單介紹了一下 Docker 的概念,演示了使用 Docker-compose 構(gòu)建全套 PHP 環(huán)境...
摘要:做了一次分享,主題使用搭建開發(fā)環(huán)境,簡單介紹了一下的概念,演示了使用構(gòu)建全套環(huán)境。應(yīng)場景通常于如下場景應(yīng)的動化打包和發(fā)布動化測試和持續(xù)集成發(fā)布在服務(wù)型環(huán)境中部署和調(diào)整數(shù)據(jù)庫或其他的后臺應(yīng)從頭編譯或者擴展現(xiàn)有的或平臺來搭建的環(huán)境。 做了一次分享,主題《使用 Docker 搭建開發(fā)環(huán)境》,簡單介紹了一下 Docker 的概念,演示了使用 Docker-compose 構(gòu)建全套 PHP 環(huán)境...
摘要:開發(fā)者在筆記本上編譯測試通過的容器可以批量地在生產(chǎn)環(huán)境中部署,包括虛擬機集群和其他的基礎(chǔ)應(yīng)用平臺。容器進(jìn)入容器名暴露端口暴露端口使用調(diào)試環(huán)境中安裝了調(diào)試,需對進(jìn)行配置后啟用,配置如下配置完成后需要重啟下容器。 showImg(https://segmentfault.com/img/bVbgmdS?w=567&h=272); Docker是一個開源的引擎,可以輕松的為任何應(yīng)用創(chuàng)建一個輕...
摘要:開發(fā)者在筆記本上編譯測試通過的容器可以批量地在生產(chǎn)環(huán)境中部署,包括虛擬機集群和其他的基礎(chǔ)應(yīng)用平臺。容器進(jìn)入容器名暴露端口暴露端口使用調(diào)試環(huán)境中安裝了調(diào)試,需對進(jìn)行配置后啟用,配置如下配置完成后需要重啟下容器。 showImg(https://segmentfault.com/img/bVbgmdS?w=567&h=272); Docker是一個開源的引擎,可以輕松的為任何應(yīng)用創(chuàng)建一個輕...
閱讀 894·2021-10-27 14:19
閱讀 1115·2021-10-15 09:42
閱讀 1538·2021-09-14 18:02
閱讀 745·2019-08-30 13:09
閱讀 2992·2019-08-29 15:08
閱讀 2093·2019-08-28 18:05
閱讀 959·2019-08-26 10:25
閱讀 2789·2019-08-23 16:28