PHP 依赖注入和控制反转再谈(二)
今天有个朋友看到yii2中介绍的依赖注入一头雾水,之前我写过类似的文章发给他看了,可能还没深入理解吧,这里我再通俗点描述下依赖注入的原理吧,尽可能滴说通俗易懂一点吧:
先还是扯下概念性滴问题(概念问题我个人的原则总是先简单瞟一眼概念,通过实例来对概念加深理解了)
要想理解 PHP 依赖注入 和 控制反转 两个概念,我们还是必须搞清楚下面的两个问题:
- DI —— Dependency Injection 依赖注入
- IoC —— Inversion of Control 控制反转
什么是依赖注入
没有工作就没有钱,没有钱我就活不下去,那么,工作和钱就是我的依赖,有钱我才能勉强滴好好的活下去,再说白了就是:
东西不是我自己的,都是我所需要的。一切需要外部提供的,都是需要进行依赖注入的。
依赖注入举例
<?php
class Work {
protected $money=5000.0;//底薪5000 public function __construct(Money $money) {
$this->money = $money;
echo 'i need money ';
}
} class Money { } $work = new Work();//输出发现
Fatal error: Uncaught TypeError: Argument 1 passed to Work::__construct() must be an instance of Money, none given, called in D:\web\test\do.php on line 15 and defined in D:\web\test\do.php:5 Stack trace: #0 D:\web\test\do.php(15): Work->__construct() #1 {main} thrown in D:\web\test\do.php on line 5
从上述代码我们可以看到Work强依赖Money必须在构造时注入Money的实例才行。我们改成如下:
<?php
class Work {
protected $money=5000.0;//底薪 public function __construct(Money $money) {
$this->money = $money;
echo 'i need money ';
}
} class Money { }
//$work = new Work();
// 所以,工作必须要给他钱才行
$money = new Money();
$work = new Work($money); //输出 i need money
那么为什么要有依赖注入这个概念,依赖注入到底解决了什么问题?我们将上述代码修正一下我们初学时都写过的代码:
<?php
class Work {
protected $money=5000.0;//底薪5000 public function __construct() {
$this->money = new Money();
}
}
这种方式与前面的方式有什么不同呢?比如某天我去了百度了底薪6000,我们会发现每次Work重生一次money.比如某天Work升职加薪到6k啦 ,怎么办?重生自己...把Money丢掉...把 6k带进去...
<?php
class BaiduMoney { } class Work {
protected $money=5000.0; public function __construct() {
// $this->money = new Money();
$this->money = new BaiduMoney();
}
}
某天 Work想去BAT....work 好烦...老换工作是不是感觉不太好?每次干不久,待的不长人却要这么的折磨自己...Work说,我要变的强大一点。我不想被改来改去的!好吧,我们让Work强大一点:
<?php
interface Money { } class baiduMoney implements Money { } class Alibaba implements Money { } class Work {
protected $money=5000.0; public function __construct(Money $money) {
$this->money = $money;
}
} $baidu= new baiduMoney();
$alibaba = new Alibaba(); $boy = new Work($baidu);
$boy = new Work($alibaba);
终于可以去BAT体验不同的人生了......
依赖注入方式
1、构造器 注入
<?php
class Book {
private $db_conn; public function __construct($db_conn) {
$this->db_conn = $db_conn;
}
}
2、setter 注入
<?php
class Book {
private $db;
private $file; function setdb($db) {
$this->db = $db;
} function setfile($file) {
$this->file = $file;
}
} class file {
} class db {
} class test {
$book = new Book();
$book->setdb(new db());
$book->setfile(new file());
}
小结:
因为大多数应用程序都是由两个或者更多的类通过彼此合作来实现业务逻辑,这使得每个对象都需要获取与其合作的对象(也就是它所依赖的对象)的引用。如果这个获取过程要靠自身实现,那么将导致代码高度耦合并且难以维护和调试。
所以才有了依赖注入的概念,依赖注入解决了以下问题:
- 依赖之间的解耦
- 单元测试,方便Mock
上面介绍的两种方法代码很清晰,但是当我们需要注入很多个依赖时,意味着又要增加很多行,会比较难以管理。
比较好的解决办法是 建立一个class作为所有依赖关系的container,在这个class中可以存放、创建、获取、查找需要的依赖关系。我们还是先来了解一下IOC的概念(又要玩概念了,不急,看代码说明一切额)
控制反转 (Inversion Of Control, IOC)
控制反转 是面向对象编程中的一种设计原则,可以用来减低计算机代码之间的耦合度。其中最常见的方式叫做 依赖注入(Dependency Injection, DI), 还有一种叫"依赖查找"(Dependency Lookup)。通过控制反转,对象在被创建的时候,由一个调控系统内所有对象的外界实体,将其所依赖的对象的引用传递给它。也可以说,依赖被注入到对象中。
<?php
class Ioc {
protected $db_conn;
public static function make_book() {
$new_book = new Book();
$new_book->set_db(self::$db_conn);
//...
//...
//其他的依赖注入
return $new_book;
}
}
此时,如果获取一个book实例,只需要执行$newone = Ioc::makebook();
以上是container的一个具体实例,最好还是不要把具体的某个依赖注入写成方法,采用registry注册,get获取比较好
<?php
/**
* 控制反转类
*/
class Ioc {
/**
* @var array 注册的依赖数组
*/
protected static $registry = array(); /**
* 添加一个 resolve (匿名函数)到 registry 数组中
*
* @param string $name 依赖标识
* @param Closure $resolve 一个匿名函数,用来创建实例
* @return void
*/
public static function register($name, Closure $resolve) {
static::$registry[$name] = $resolve;
} /**
* 返回一个实例
*
* @param string $name 依赖的标识
* @return mixed
* @throws \Exception
*/
public static function resolve($name) {
if (static::registered($name)) {
$name = static::$registry[$name];
return $name();
} throw new \Exception("Nothing registered with that name");
} /**
* 查询某个依赖实例是否存在
*
* @param string $name
* @return bool
*/
public static function registered($name) {
return array_key_exists($name, static::$registry);
}
}
现在就可以通过如下方式来注册和注入一个
<?php
Ioc::register("book", function () {
$book = new Book();
$book->setdb('db');
$book->setfile('file'); return $book;
}); // 注入依赖
$book = Ioc::resolve('book');
问题汇总
1、参与者都有谁?
答:一般有三方参与者,一个是某个对象;一个是IoC/DI的容器;另一个是某个对象的外部资源。又要名词解释一下,某个对象指的就是任意的、普通的Java对象; IoC/DI的容器简单点说就是指用来实现IoC/DI功能的一个框架程序;对象的外部资源指的就是对象需要的,但是是从对象外部获取的,都统称资源,比如:对象需要的其它对象、或者是对象需要的文件资源等等。
2、依赖:谁依赖于谁?为什么会有依赖?
答:某个对象依赖于IoC/DI的容器。依赖是不可避免的,在一个项目中,各个类之间有各种各样的关系,不可能全部完全独立,这就形成了依赖。传统的开发是使用其他类时直接调用,这会形成强耦合,这是要避免的。依赖注入借用容器转移了被依赖对象实现解耦。
3、注入:谁注入于谁?到底注入什么?
答:通过容器向对象注入其所需要的外部资源
4、控制反转:谁控制谁?控制什么?为什么叫反转?
答:IoC/DI的容器控制对象,主要是控制对象实例的创建。反转是相对于正向而言的,那么什么算是正向的呢?考虑一下常规情况下的应用程序,如果要在A里面使用C,你会怎么做呢?当然是直接去创建C的对象,也就是说,是在A类中主动去获取所需要的外部资源C,这种情况被称为正向的。那么什么是反向呢?就是A类不再主动去获取C,而是被动等待,等待IoC/DI的容器获取一个C的实例,然后反向的注入到A类中。
5、依赖注入和控制反转是同一概念吗?
答:从上面可以看出:依赖注入是从应用程序的角度在描述,可以把依赖注入描述完整点:应用程序依赖容器创建并注入它所需要的外部资源;而控制反转是从容器的角度在描述,描述完整点:容器控制应用程序,由容器反向的向应用程序注入应用程序所需要的外部资源。
PHP 依赖注入和控制反转再谈(二)的更多相关文章
- 浅谈(IOC)依赖注入与控制反转(DI)
前言:参考了百度文献和https://www.cnblogs.com/liuqifeng/p/11077592.html以及http://www.cnblogs.com/leoo2sk/archive ...
- 轻松学,浅析依赖倒置(DIP)、控制反转(IOC)和依赖注入(DI) 依赖注入和控制反转的理解,写的太好了。
轻松学,浅析依赖倒置(DIP).控制反转(IOC)和依赖注入(DI) 2017年07月13日 22:04:39 frank909 阅读数:14269更多 所属专栏: Java 反射基础知识与实战 ...
- Spring Framework------>version4.3.5.RELAESE----->Reference Documentation学习心得----->Spring Framework的依赖注入和控制反转
Dependency Injection and Inversion of Control 1.概述: 1.1相关概念 bean:由IoC容器所管理的对象,也即各个类实例化所得对象都叫做bean 控制 ...
- 简单解析依赖注入(控制反转)在Spring中的应用
IoC——Inversion of Control 控制反转DI——Dependency Injection 依赖注入 大家都知道,依赖注入是Spring中非常重要的一种设计模式.可能很多初学者 ...
- 大话DI依赖注入+IOC控制反转(一) 之 定义
转发时请注明原创作者及地址,否则追究责任.原创:alunchen 依赖注入与控制反转 依赖注入与控制反转是老生常谈的问题.一般面试也会面试到这种问题.网上很多很多这方面的资料,搜索出来一大堆 ...
- Git使用总结 Asp.net生命周期与Http协议 托管代码与非托管代码的区别 通过IEnumerable接口遍历数据 依赖注入与控制反转 C#多线程——优先级 AutoFac容器初步 C#特性详解 C#特性详解 WPF 可触摸移动的ScrollViewer控件 .NET(C#)能开发出什么样的APP?盘点那些通过Smobiler开发的移动应用
一,原理 首先,我们要明白Git是什么,它是一个管理工具或软件,用来管理什么的呢?当然是在软件开发过程中管理软件或者文件的不同版本的工具,一些作家也可以用这个管理自己创作的文本文件,由Linus开发的 ...
- ADO.NET .net core2.0添加json文件并转化成类注入控制器使用 简单了解 iTextSharp实现HTML to PDF ASP.NET MVC 中 Autofac依赖注入DI 控制反转IOC 了解一下 C# AutoMapper 了解一下
ADO.NET 一.ADO.NET概要 ADO.NET是.NET框架中的重要组件,主要用于完成C#应用程序访问数据库 二.ADO.NET的组成 ①System.Data → DataTable, ...
- 谈谈php依赖注入和控制反转
要想理解php依赖注入和控制反转两个概念,就必须搞清楚如下的问题: DI--Dependency Injection 依赖注入 IoC--Inversion of Control 控制反转 1. ...
- Java的依赖注入(控制反转)
两个主角"依赖注入"和"控制反转": 1.二都说的都是同一件事,只是叫法不同.是一个重要的面向对象编程的法则,也是一种设计模式: 2.英文原称:依赖注入,Dep ...
随机推荐
- beeline方式连接hive
什么是beeline 常用操作 https://www.cnblogs.com/xinfang520/p/7684598.html
- spring mvc jsonp调用示例
服务端代码:主要是返回的时候,返回值要用callback包装一下 /** * JSONP调用 * * @param request * @return */ @RequestMapping(" ...
- Dubbo 源码分析 - 集群容错之 Cluster
1.简介 为了避免单点故障,现在的应用至少会部署在两台服务器上.对于一些负载比较高的服务,会部署更多台服务器.这样,同一环境下的服务提供者数量会大于1.对于服务消费者来说,同一环境下出现了多个服务提供 ...
- FastDFS客户端与自定义文件存储系统
<1>安装 安装提供给大家的fdfs_client-py-master.zip到虚拟环境中 pip install fdfs_client-py-master.zip pip instal ...
- JavaScript变量与数据类型详解
变量 变量来源于数学,是计算机语言中能储存计算结果或能表示值抽象概念.变量可以通过变量名访问. 变量的作用就是用于存储值. 语法: 声明变量时,总是以关键字var打头.任何情况下都应该这样做.然后给变 ...
- Spring 通知和顾问进行增强
使用顾问增加前置增强和后置增强 <bean id="1" class="目标对象"></bean> <bean id=" ...
- 排除Transformation Errors
当运行session时,会产生大量的Transformation Errors,这些Error会导致性能变慢 1 Transformation Errors导致性能降低的原因 当有大量记录有Trans ...
- 阿里云申请ssl证书
申请证书(本文以阿里云服务器为背景,申请证书也以阿里云域名申请证书来作为实例) (1)登陆阿里云服务器,初次配置的用户,不建议直接搜索‘ssl证书’进行购买,因为这样购买后证书与域名对应的引导性并不强 ...
- foreach加循环体与不加循环体的区别
案例A(不加{}): <?php function genTree5($items) { foreach ($items as $item) echo $item['id'];die; $ite ...
- Python快速学习08:模块的操作
前言 系列文章:[传送门] 天气干燥,我就上火,流鼻血.希望身子好起来. 正文 函数和对象都是为了更好的组织已经有的程序,以方便重复利用. 模块(module)也是为了同样的目的.模块可以包含可执行代 ...