很多初入职场的PHP新人或者做了几年CURD的同学,还是一头雾水,何谓DI?这个问题熟悉Java等编程语言的略过。
什么是依赖注入 (DI)
依赖注入(DI)的概念虽然听起来很深奥,但是如果你用过一些新兴的php框架的话,对于DI一定不陌生,因为它们多多少少都用到了依赖注入来处理类与类之间的依赖关系。
- 依赖注入 (DI)其实本质上是指对类的依赖通过构造器完成自动注入。
- 通俗来说,就是你当前操作一个类,但是这个类的某些方法或者功能不是单单只靠这个类就能完成的,而是要借助另一个类的才能完成的。
- 最直接的标志就是传参数据为对象的时候。严格来说,你想在另一个类中操作另一个类,这两个类之间形成了相互依赖关系,传参的方式叫注入。
举例说明:
控制器:
<?php
namespace app\index\controller;
use app\index\model\User;
class Debug
{
protected $user;
public function __construct(User $user)
{
$this->user = $user;
}
public function hello()
{
return 'Hello,' . $this->user->name . '!';
}
}
模型:
<?php
namespace app\index\model;
use think\Model;
class User extends Model
{
public $name = '我是依赖注入';
}
运行 app\index\index\debug\hello:Hello,我是依赖注入!
依赖注入出现的原因
- 在刚开始的时候,php需要在一个类中使用另一个类的时候,都会如下操作
- 比如我在container类中需要用到adapter类,就需要在使用之前进行实例化
- 如果需要用到大量的外部类,这就会造成了耦合度太高,很容易造成后期的维护困难
- 通俗地来讲,也就是container脱离不了外部类去工作,这就叫耦合度太高
<?php
class container
{
private $adapter;
public function __construct()
{
$this->adapter = new adapter();
}
}
简单的依赖注入
- 上面的代码耦合度太高,导致了依赖注入的出现,主要是为了解耦合
- 如下图,我们只需要将所需要操作的类对象传入即可
- 依赖注入操作的参数是对象,而不是普通参数,是不是有更好的理解了
- 但是这样的简单依赖注入,会造成如果你依赖的类很多,你传参的时候会很长,容易混乱
一个简单的依赖注入的例子
class Container {
private $s=array();
function __set($k, $c) { $this->s[$k]=$c; }
function __get($k) { return $this->s[$k]($this); }
}
有了container类之后我们可以怎样管理A与B之间的依赖关系呢,用代码说话吧:
class A
{
private $container;
public function __construct(Container $container)
{
$this->container = $container;
}
public function doSomeThing()
{
//do something which needs class B
$b = $this->container->getB();
//to do
}
}
再将B类注入到容器类中:
$c = new Container();
$c->setB(new B());
还可以传入一个匿名函数,这样B类就不会在传入时就立即实例化,而是在真正调用时才完成实例化的工作:
$c = new Container();
$c->setB(function (){
return new B();
});
高阶的依赖注入
- 为了解决上面参数混乱的问题,这时候,依赖注入进行了进化
- 通过魔术方法,__get去设置对象
- 这时候,我们就可以解决依赖太多,参数混乱的问题了
<?php
class container
{
private $adapter;
public function __construct(adapter $adapter)
{
$this->adapter = $adapter;
}
}
依赖注入的应用
- 我们先定义一个容器类,主要用来向容器中注入你想要操作的类
- 使用的时候,只需要传容器这一个对象即可
<?php
class container
{
public $instance = [];
public function __set($name, $value)
{
$this->instance[$name] = $value;
}
}
class adapter
{
public $name = '我是调度器';
}
$container = new container();
$container->adapter = new adapter();
class autofelix
{
private $container;
public function __construct(container $container)
{
$this->container = $container;
}
public function who($class)
{
return $this->container->instance[$class]->name;
}
}
$autofelix = new autofelix($container);
$who = $autofelix->who('adapter');
var_dump($who); //我是调度器
高阶优化
- 在上面的应用中,我们直接将实例化后的对象注入容器中
- 这样会导致,所有的对象还没有被使用就会被实例化一遍,造成资源的损耗
- 我们可以传入闭包,这样对象就不会被实例化而注入,当你自己需要使用的时候,再去实例化
- 就可以减少服务器资源的损耗了
<?php
$container = new container();
$container->adapter = new adapter();
//高阶优化
$container = new container();
$container->adapter = function () {
return new adapter();
};