依赖注入(Dependence Injection, DI) 依赖注入是控制反转的一种设计模式。依赖注入的核心是把类所依赖的单元的实例化过程,放到类的外面去实现。依赖注入的实现离不开反射。
依赖注入(Dependence Injection, DI)
所谓的依赖注入,指将依赖的对象通过参数的形式一次性传入,使用时不需要显式 new
了,比如把A类所依赖的B类、C类等以属性或者构造函数等方式注入A类而不是直接在A类中实例化。
只要不是由内部生产(比如初始化、构造函数中通过工厂方法、自行手动 new 的),而是由外部以参数或其他形式注入的,都属于依赖注入(DI) 。
依赖注入需要利用反射实现,比如:
class A { protected $b; public function __constrcut(B $b) { $this->b = $b; } } // 通过控制反转容器生成 A 的实例时,会通过反射发现 A 的构造函数需要一个 B 类的实例 // 于是自动向 A 类的构造函数注入 B 类的实例 $a = IoC::make(A::class);
依赖注入的实质就是把一个类不可更换的部分
和可更换的部分
分离开来,通过注入
的方式来使用,从而达到解耦的目的。
比如有一个 Mysql
数据库连接类如下:
class Mysql { private $host; private $port; private $username; private $password; private $db_name; public function __construct(){ $this->host = '127.0.0.1'; $this->port = 22; $this->username = 'root'; $this->password = ''; $this->db_name = 'db'; } public function connect() { return mysqli_connect($this->host, $this->username, $this->password, $this->db_name, $this->port); } } // 使用 $db = new Mysql(); $con = $db->connect();
可更换部分是数据库的配置
class Configuration { private $host; private $port; private $username; private $password; private $db_name; public function __construct($host, $port, $username, $password, $db_name) { $this->host = $host; $this->port = $port; $this->username = $username; $this->password = $password; $this->db_name = $db_name; } public function getHost() { return $this->host; } public function getPort() { return $this->port; } public function getUsername() { return $this->username; } public function getPassword() { return $this->password; } public function getDbName() { return $this->db_name; } }
不可更换部分是Mysql数据库的连接操作
class Mysql { private $configuration; public function __construct(Configuration $config) { $this->configuration = $config; } public function connect(){ return mysqli_connect($this->configuration->getHost(),$this->configuration->getUsername() , $this->configuration->getPassword,$this->configuration->getDbName(),$this->configuration->getPort()); } }
这样就完成了配置文件和连接逻辑的分离,使用如下:
$config = new Configuration('127.0.0.1', 'root', '', 'my_db', 22); $db = new Mysql($config); $con = $db->connect();
$config是注入Mysql的,这就是所谓的依赖注入。
总结
注入可以理解成从外面把东西打进去。因此,依赖注入模式中,要分清内和外,要解除依赖的类内部就是内,实例化所依赖单元的地方就是外。在外通过构造形参,为类内部的抽象单元提供实例化,达到解耦的目的,使下层依赖于上层,而不是上层依赖于下层。
因此,依赖注入模式中,要用对象传递,通过一个实例的反射来实现,这是数组做不到的。
来源:https://www.cnblogs.com/benbenhan/p/12376802.html