如何利用PHP制作轻量级容器,过程步骤是什么
Admin 2022-07-06 群英技术资讯 772 次浏览
这篇文章主要讲解了“如何利用PHP制作轻量级容器,过程步骤是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何利用PHP制作轻量级容器,过程步骤是什么”吧!什么是容器
在开发过程中,经常会用到的一个概率就是依赖注入。我们借助依懒注入来解耦代码,选择性的按需加载服务,而这些通常都是借助容器来实现。
容器实现对类的统一管理,并且确保对象实例的唯一性
常用的容器网上有很多,如PHP-DI 、 YII-DI 等各种实现,通常他们要么大而全,要么高度适配特定业务,与实际需要存在冲突。
出于需要,我们自己造一个轻量级的轮子,为了保持规范,我们基于PSR-11 来实现。
PSR-11
PSR 是 php-fig 提供的标准建议,虽然不是官方组织,但是得到广泛认可。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);
}
实现示例
我们先来实现接口中要求的两个方法
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]);
}
实际我们容器中注入的对象是多种多样的,所以我们单独抽出实例化方法。
public function make($name)
{
if (!is_string($name)) {
throw new \InvalidArgumentException(sprintf(
'The name parameter must be of type string, %s given',
is_object($name) ? get_class($name) : gettype($name)
));
}
if (isset($this->resolvedEntries[$name])) {
return $this->resolvedEntries[$name];
}
if (!$this->has($name)) {
throw new NotFoundException("No entry or class found for {$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;
}
/**
*
* 获取构造类相关参数的依赖
* @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;
}
如你所见,到目前为止我们只实现了从容器中取出实例,从哪里去提供实例定义呢,所以我们还需要提供一个方水法
/**
* @param string $id
* @param string | array | callable $concrete
* @throws ContainerException
*/
public function injection($id, $concrete)
{
if (is_array($concrete) && !isset($concrete['class'])) {
throw new ContainerException('数组必须包含类定义');
}
$this->definitions[$id] = $concrete;
}
只有这样吗?对的,有了这些操作我们已经有一个完整的容器了,插箱即用。
不过为了使用方便,我们可以再提供一些便捷的方法,比如数组式访问。
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]);
}
}
这样我们就拥有了一个功能丰富,使用方便的轻量级容器了,赶快整合到你的项目中去吧。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。
猜你喜欢
会有Notice:Undefinedvariableecho$myvariable+3;//使用这个变量出现:Notice:Undefinedvariable:myvariablein如果将其改写成。
冒泡排序就如其名字一样,轻的气泡在上面,重的在下面。原理:对要排序的一列数作两两交换,最小的在最左边,每次都能在剩下的数中找出最小的数,将冒出来的这些数组成个有序的排列,剩下的无序排列,有序的都比无序的小。
php中get_object_vars()在数组的使用:1、获取$object对象中的属性,组成一个数组;2、语法,get_object_var($object);3、使用注意,缺点,只转一维,不会递归。
本特性可以使用户上传文本和二进制文件。用PHP的认证和文件操作函数,可以完全控制允许哪些人上传以及文件上传后怎样处理。PHP能够接受任何来自符合RFC-1867标准的浏...
php接口中interface的使用方法:1、说明,接口中的方法都是抽象方法,并且是public;2、接口同样可以继承接口,但是子类必须实现所有方法;3、实现多重继承。
成为群英会员,开启智能安全云计算之旅
立即注册关注或联系群英网络
7x24小时售前:400-678-4567
7x24小时售后:0668-2555666
24小时QQ客服
群英微信公众号
CNNIC域名投诉举报处理平台
服务电话:010-58813000
服务邮箱:service@cnnic.cn
投诉与建议:0668-2555555
Copyright © QY Network Company Ltd. All Rights Reserved. 2003-2020 群英 版权所有
增值电信经营许可证 : B1.B2-20140078 粤ICP备09006778号 域名注册商资质 粤 D3.1-20240008