当前位置:网站首页>一文搞懂│php 中的 DI 依赖注入
一文搞懂│php 中的 DI 依赖注入
2022-08-03 18:23:00 【极客飞兔】
什么是 DI / 依赖注入
依赖注入DI
其实本质上是指对类的依赖通过构造器完成自动注入
- 通俗来说,就是你当前操作一个类,但是这个类的某些方法或者功能不是单单只靠这个类就能完成的,而是要
借助另一个类
的才能完成的 - 最直接的标志就是传参数据为对象的时候。严格来说,你
想在一个类中操作另一个类
,这两个类之间形成了相互依赖关系,传参的方式叫注入
依赖注入出现的原因
- 在未使用依赖注入的时候,
php
需要在一个类中使用另一个类的时候,往往都会进行如下操作 - 比如我在
container
类中需要用到adapter
类,就需要在使用之前进行实例化 - 如果需要用到大量的外部类,这就会造成了
耦合度太高
,很容易造成后期的维护困难
- 通俗的来讲,也就是
container
脱离不了外部类去工作,这就叫耦合度太高
<?php
class container
{
private $adapter;
public function __construct()
{
$this->adapter = new adapter();
}
}
简单的依赖注入
- 上面的代码耦合度太高,导致了
依赖注入
的出现,主要是为了解耦合
- 如下案例,我们只需要将所需要操作的类对象传入即可
依赖注入
操作的参数是对象
,而不是普通参数,是不是有更好的理解了- 但是这样的简单依赖注入,会造成如果你依赖的类很多,你传参的时候会很长,容易
混乱
<?php
class container
{
private $adapter;
public function __construct(adapter $adapter)
{
$this->adapter = $adapter;
}
}
高阶的依赖注入
- 为了解决上面
参数混乱
的问题,这时候,依赖注入进行了优化
通过魔术方法,__get
去设置对象 - 这时候,我们就可以解决依赖太多,参数混乱的问题了
<?php
class container
{
public $instance = [];
public function __set($name, $value)
{
$this->instance[$name] = $value;
}
}
$container = new container();
$container->adapter = new adapter();
$container->autofelix = new autofelix();
依赖注入的应用
- 我们先定义一个
容器类
,主要用来向容器中注入
你想要操作的类 - 使用的时候,只需要传容器这一个
对象
即可
<?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();
};
边栏推荐
- 广告电商、泰山众筹、链动2+1,这3个模式到底怎么样?
- es6新增-Promise详解(异步编程的解决方案1)
- Flask框架——项目可安装化
- Shell:循环语句
- 超T动力 焕“芯”出发 | 中国重汽专属定制版WP14T产品闪耀登场
- InnoDB 中不同SQL语句设置的锁
- WEB 渗透之RCE
- Big guy, who is free to help me to see what the problem is, I just read MySQL source print, and I just came into contact with flink.
- Execution plan of mysql
- 六、用户身份认证
猜你喜欢
【美丽天天秒】链动2+1模式开发
三丁基-巯基膦烷「tBuBrettPhos Pd(allyl)」OTf),1798782-17-8
【汇编语言03】第2章 寄存器——实验1:查看CPU和内存,用机器指令和汇编指令编程
[Azure Event Hub] Create Event Hub Consume Client + Custom Event Position with Azure AD Authentication
首届MogDB征文活动开启啦!
87.(cesium之家)cesium热力图(贴地形)
es6新增-Generator(异步编程的解决方案2)
H.265网页播放器EasyPlayer获取视频流正常,但是播放出现黑屏是什么原因?
全尺度表示的上下文非局部对齐,南科大&优图提出NAFS解决基于文本的Re ID
rhel8.3 系统下修改有线网卡配置信息实现联网
随机推荐
Jenkins CI平台(二)
Oracle备份的几种方式
大佬们,flinkcdc 2.2 版本采集sqlserver只能采集到全量的数据,不能采集到增量的数
实时渲染器不止lumion,Chaos Vantage你值得一试
TiFlash 计算层概览
USD 能统一元宇宙吗?
Execution plan of mysql
架构基本概念和架构本质
ImportError: /lib/libgdal.so.26: undefined symbol: sqlite3_column_table_name
异常与智能指针
LyScript 内存交换与差异对比
fatal error: jni.h: No such file or directory
有人知道flink sql 使用tableEnv.executeSql执行后,怎么获取到任务运行的
Unable to start SinkRunner: { policy:org.apache.flume
WEB 渗透之RCE
5000元价位高性能轻薄本标杆 华硕无双高颜能打
微信小程序分享功能
Chrome浏览器开发新截图工具,安全浏览器截图方法
开篇-开启全新的.NET现代应用开发体验
MPLS的简单应用于实验