文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

怎么创建PHP DI容器

2023-06-21 22:21

关注

这篇文章主要讲解了“怎么创建PHP DI容器”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么创建PHP DI容器”吧!

由开车开始

先开个车,为大家举个栗子:

class Driver{    public function drive()    {        $car = new Car();        echo '老司机正在驾驶', $car->getCar(), PHP_EOL;    }}class Car{    protected $name = '普通汽车';    public function getCar()    {        return $this->name;    }}

有两个类,Driver和Car,老司机Driver有个方法driver,在调用的时候首先得整辆车$car,然后发车。大多数同学都写过这样或者类似的代码,这样的代码单看没啥毛病,挺正常的。但是,如果我要换辆车,开普通车撩不到妹。

class Benz extends Car{    protected $name = '奔驰';}

这时候就需要做一个比较恶心的操作了,得改老司机的代码了。(老司机:我做错了什么?换辆车还得让我重学驾照……)。因此我们需要把让Car为外界注入,将Driver和Car解耦,不是老司机自己开车的时候还得自己去造车。于是就有了下面的结果

class Driver{    protected $car;    public function __construct(Car $car)    {        $this->car = $car;    }    public function drive()    {        echo '老司机正在驾驶', $this->car->getCar(), PHP_EOL;    }}

此时Driver和Car两个类已经解耦,这两个类的依赖,依靠上层代码去管理。此时,老司机会这样“开车”:

$car = new Car();$driver = new Driver($car);$driver->drive();

此时,我们创建Driver依赖的实例,并注入。上面的例子,我们实现了依赖注入,不过是手动的,写起来感觉还是不爽。这么繁重的活怎么能手动来做呢,得让程序自己去做。于是乎,DI容器诞生。

依赖注入容器

依赖注入与IoC模式类似工厂模式,是一种解决调用者和被调用者依赖耦合关系的模式。它解决了对象之间的依赖关系,使得对象只依赖IoC/DI容器,不再直接相互依赖,实现松耦合,然后在对象创建时,由IoC/DI容器将其依赖(Dependency)的对象注入(Inject)其内,这样做可以最大程度实现松耦合。依赖注入说白一点,就是容器将某个类依赖的其他类的实例注入到这个类的实例中。

这段话可能说的有点抽象,回到刚才的例子吧。刚刚我手动完成了依赖注入,比较麻烦,如果一个大型的项目这样做肯定会觉得很繁琐,而且不够优雅。因此我们需要有一位总管代替我们去干这个,这个总管就是容器。类的依赖管理全部交给容器去完成。因此,一般来说容器是一个全局的对象,大家共有的。

做一个自己的DI容器

写一个功能,我们首先需要分析问题,因此我们先要明白,对于一个简单的DI容器需要哪些功能,这直接关系到我们代码的编写。对于一个简单的容器,至少需要满足以下几点:

综上,我们的容器类大约长这样:

class Container{        protected static $instance;        protected $instances = [];    private function __construct(){}      private function __clone(){}        public function singleton($class, ...$params)    {}        public function get($class, ...$params)    {}        protected function make($class, $params = [])    {}        public static function getInstance()    {        if (null === static::$instance) {            static::$instance = new static();        }        return static::$instance;    }}

大体骨架已经确定,接下来进入最核心的make方法:

protected function make($class, $params = []){  //如果不是反射类根据类名创建  $class = is_string($class) ? new ReflectionClass($class) : $class;  //如果传的入参不为空,则根据入参创建实例  if (!empty($params)) {    return $class->newInstanceArgs($params);  }  //获取构造方法  $constructor = $class->getConstructor();  //获取构造方法参数  $parameterClasses = $constructor ? $constructor->getParameters() : [];  if (empty($parameterClasses)) {    //如果构造方法没有入参,直接创建    return $class->newInstance();  } else {    //如果构造方法有入参,迭代并递归创建依赖类实例    foreach ($parameterClasses as $parameterClass) {      $paramClass = $parameterClass->getClass();      $params[] = $this->make($paramClass);    }    //最后根据创建的参数创建实例,完成依赖的注入    return $class->newInstanceArgs($params);  }}

为了容器的易用,我做了一些完善:

最终版:

class Container implements ArrayAccess{        protected static $instance;        protected $instances = [];    private function __construct(){}    private function __clone(){}        public function singleton($class, ...$params)    {        if (isset($this->instances[$class])) {            return $this->instances[$class];        } else {            $this->instances[$class] = $this->make($class, $params);        }        return $this->instances[$class];    }        public function get($class, ...$params)    {        return $this->make($class, $params);    }        protected function make($class, $params = [])    {        //如果不是反射类根据类名创建        $class = is_string($class) ? new ReflectionClass($class) : $class;        //如果传的入参不为空,则根据入参创建实例        if (!empty($params)) {            return $class->newInstanceArgs($params);        }        //获取构造方法        $constructor = $class->getConstructor();        //获取构造方法参数        $parameterClasses = $constructor ? $constructor->getParameters() : [];        if (empty($parameterClasses)) {            //如果构造方法没有入参,直接创建            return $class->newInstance();        } else {            //如果构造方法有入参,迭代并递归创建依赖类实例            foreach ($parameterClasses as $parameterClass) {                $paramClass = $parameterClass->getClass();                $params[] = $this->make($paramClass);            }            //最后根据创建的参数创建实例,完成依赖的注入            return $class->newInstanceArgs($params);        }    }        public static function getInstance()    {        if (null === static::$instance) {            static::$instance = new static();        }        return static::$instance;    }    public function __get($class)    {        if (!isset($this->instances[$class])) {            $this->instances[$class] = $this->make($class);        }        return $this->instances[$class];    }    public function offsetExists($offset)    {        return isset($this->instances[$offset]);    }    public function offsetGet($offset)    {        if (!isset($this->instances[$offset])) {            $this->instances[$offset] = $this->make($offset);        }        return $this->instances[$offset];    }    public function offsetSet($offset, $value)    {    }    public function offsetUnset($offset) {        unset($this->instances[$offset]);    }}

现在借助容器我们写一下上面的代码:

$driver = $app->get(Driver::class);$driver->drive();//output:老司机正在驾驶普通汽车复制代码

就这么简单,老司机就能发车。这里默认注入的是Car的实例,如果需要开奔驰,那只需要这样:

$benz = $app->get(Benz::class);$driver = $app->get(Driver::class, $benz);$driver->drive();//output:老司机正在驾驶奔驰复制代码

按照PSR-11的要求,依赖注入容器需要实现Psr\Container\ContainerInterface接口,这里只是演示并未去实现,因为那需要引入Psr依赖库,比较麻烦,其实也很简单,只是多了几个方法,有兴趣的可以自己去了解下PSR-11的要求(传送门)。

这里只是实现了一个非常简陋的DI容器,实际中还需要考虑很多,而且这里的容器功能上还很简陋。还有一些坑没处理,比如出现循环依赖怎么处理、延迟加载的机制……

感谢各位的阅读,以上就是“怎么创建PHP DI容器”的内容了,经过本文的学习后,相信大家对怎么创建PHP DI容器这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯