PHP面向对象五大原则之依赖倒置原则(DIP)详解


Posted in PHP onApril 08, 2018

本文实例讲述了PHP面向对象五大原则之依赖倒置原则(DIP)。分享给大家供大家参考,具体如下:

什么是依赖倒置呢?简单地讲就是将依赖关系倒置为依赖接口,具体概念如下:

1.上层模块不应该依赖于下层模块,它们共同依赖于一个抽象(父类不能依赖子类,它们都要依赖于抽象类)

2.抽象不能依赖于具体,具体应该要依赖于抽象。

注意,这里的接口不是狭义的接口

为什么要依赖接口?因为接口体现对问题的抽象,同时由于抽象一般是相对稳定的或者是相对变化不频繁的,而具体是易变的。因此依赖抽象是实现代码扩展和运行期内绑定(多态)的基础:只要实现了该抽象类的子类,都可以被类的使用都使用。这里,强调一下扩展性这个概念。通常扩展性指对已知行为的扩展,在讲述接口时,也提到过,接口应该是相对的。这就告诉我们,无论使用多么先进的设计模式,也无法做到不需要修改代码即可达到不变应万变的地上。在面向对象的这五大原则里,我认为依赖倒置是最难理解,也是最难实现的。

这里以雇员类为例

<?php
interface employee
{
  public function working();
}
class teacher implements employee
{
  public function working()
  {
    echo 'teaching...';
  }
}
class coder implements employee
{
  public function working()
  {
    echo 'coding...';
  }
}
class workA
{
  public function work()
  {
    $teacher = new teacher();
    $teacher->working();
  }
}
class workB
{
  private $e;
  public function set(employee $e)
  {
    $this->e = $e;
  }
  public function work()
  {
    $this->e->working();
  }
}
$worka = new workA;
$worka->work();
$workb = new workB;
$workb->set(new teacher());
$workb->work();

在workA中,work方法依赖于teacher实现;在workB中,work转而依赖抽象,这样可以把需要的对象通过参数传入。上述代码通过接口,实现了一定程度的解耦,但仍然是有限的。不仅是使用接口,使用工厂等也能实现一定程度的解耦和依赖倒置。

在workB中,teacher实例通过set方法传入,从而实现了工厂模式。由于这样的实现仍然是硬编码的,为了实现代码的进一步扩展,把这个依赖关系写在配置文件里,指明workB需要一个teacher对象,专门由一个程序配置是否正确(如所依赖的类文件是否存在)以及加载配置中所依赖的实现,这个检测程序,就称为IOC容器。

很多文章里看到IOC(Inversion of Control)概念,实际上,IOC是依赖倒置原则(Dependence Inversion Principle,DIP)的同义词。而在提IOC的时候,你可能还会看到有人提起DI等概念。DI,即依赖注入,一般认为,依赖注入(DI)和依赖查找(DS)是IOC的两种实现。不过随着某些概论的演化,这几个概念之间的关系也变得很模糊,也有人认为IOC就是DI。有人认为,依赖注入的描述比起IOC来更贴切,这里不纠缠于这几个概念之间的关系。

在经典的J2EE设计里,通常把DAO层和Servicen层细分为接口层和实现层,然后在配置文件里进行所依赖关系的配置,这是最常见的DIP的应用。Spring框架就是一个很好的IOC容器,把控制权从代码剥离到IOC窗口,这里是通过XML配置文件实现的,Spring在执行期间根据配置文件的设定,建立对象之间的依赖关系。

如下面的代码所示

<bean scopre="prototype" class="cn.notebook.action.NotebookListOtherAction" id="notebookListOtherAction">
  <property ref="userReplyService" name="userReplyService" />
  <property ref="userService" name="userService" />
  <property ref="permissionService" name="permissionService" />
  <property ref="friendService" name="friendService" />
</bean>

但是这样的设置一样存在问题,配置文件会变得越来越大,其间关系会越来越复杂。同样逃脱不了随着应用和业务的改变,不断修改代码的恶魇(这里认为配置文件是代码的一部分。并且在实际开发中,很少存在单纯修改配置文件的情况。一般配置文件修改了,代码也会做相应的修改)

在PHP里,也有类似模仿Spring的实现,即把依赖关系写在了配置文件里,通过配置文件来产生需要的对象。我觉得这样的代码还是为了实现而实现。在Srping里,配置文件里配置的不仅仅是一个类运行时的依赖关系,还可以实现事务管理、AOP、延迟加载等。而PHP要实现上面的种种特性,其消耗是巨大的。从语言层面讲,PHP这种动态脚本语言在实现一些多态特性上和编译型的语言不同。其次PHP作为敏捷性的开发语言,更强调快速开发、逻辑清晰、代码更简单易懂,如果再附加了各种设计模式的框架,从技术实现和运行效率上来看,都是不可取的。依赖倒置的核心原则是解耦。如果脱离这个最原始的原则,那就是本末倒置。

事实上,很多的设计模式里已经隐含了依赖倒置原则我们也在有意无意地做着一些依赖反转的工作。只是作为PHP,目前还没有一个比较完善的IOC容器,或许是PHP根本不需要。

如果满足DIP:

1.每个较高层次类都为它所需要的服务提出一个接口声明,较低层次类实现实现这个接口。

2.每个高层次类都通过该抽象接口使用服务。

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
如何正确理解PHP的错误信息
Oct 09 PHP
php4的session功能评述(二)
Oct 09 PHP
php 动态添加记录
Mar 10 PHP
实测在class的function中include的文件中非php的global全局环境
Jul 15 PHP
基于php和mysql的简单的dao类实现crud操作功能
Jan 27 PHP
php更新修改excel中的内容实例代码
Feb 26 PHP
PHP中模拟链表和链表的基本操作示例
Feb 27 PHP
Ubuntu server 11.04安装memcache及php使用memcache来存储session的方法
May 31 PHP
使用Codeigniter重写insert的方法(推荐)
Mar 23 PHP
PHPExcel中文帮助手册|PHPExcel使用方法(分享)
Jun 09 PHP
laravel 5.4中实现无限级分类的方法示例
Jul 27 PHP
解决Laravel5.2 Auth认证退出失效的问题
Oct 14 PHP
PHP面向对象五大原则之里氏替换原则(LSP)详解
Apr 08 #PHP
PHP实现微信红包金额拆分试玩的算法示例
Apr 07 #PHP
PHP面向对象五大原则之接口隔离原则(ISP)详解
Apr 04 #PHP
PHP面向对象五大原则之开放-封闭原则(OCP)详解
Apr 04 #PHP
PHP面向对象五大原则之单一职责原则(SRP)详解
Apr 04 #PHP
PHP基于面向对象实现的留言本功能实例
Apr 04 #PHP
PHP设计模式之工厂模式定义与用法详解
Apr 03 #PHP
You might like
详解PHP数据压缩、加解密(pack, unpack)
2016/12/17 PHP
简单谈谈PHP中的trait
2017/02/25 PHP
40款非常有用的 jQuery 插件推荐(系列一)
2011/12/21 Javascript
JavaScript字符串对象的concat方法实例(用于连接两个或多个字符串)
2014/10/16 Javascript
jQuery实现防止提交按钮被双击的方法
2015/03/24 Javascript
JavaScript实现MIPS乘法模拟的方法
2015/04/17 Javascript
javascript中return,return true,return false三者的用法及区别
2015/11/17 Javascript
Javascript自执行匿名函数(function() { })()的原理浅析
2016/05/15 Javascript
jQuery EasyUI提交表单验证
2016/07/19 Javascript
HTML5+jQuery实现搜索智能匹配功能
2017/03/24 jQuery
在iframe中使bootstrap的模态框在父页面弹出问题
2017/08/07 Javascript
Vue iview-admin框架二级菜单改为三级菜单的方法
2018/07/03 Javascript
vue框架中props的typescript用法详解
2020/02/17 Javascript
Vue实现购物车实例代码两则
2020/05/30 Javascript
国内常用的js类库大全(CDN公共库)
2020/06/24 Javascript
vue v-model的用法解析
2020/10/19 Javascript
解决ant design vue 表格a-table二次封装,slots渲染的问题
2020/10/28 Javascript
Bootstrap FileInput实现图片上传功能
2021/01/28 Javascript
Python socket C/S结构的聊天室应用实现
2014/11/30 Python
python 全局变量的import机制介绍
2017/09/07 Python
R vs. Python 数据分析中谁与争锋?
2017/10/18 Python
使用pycharm在本地开发并实时同步到服务器
2019/08/02 Python
Django中使用session保持用户登陆连接的例子
2019/08/06 Python
详解Python的三种拷贝方式
2020/02/11 Python
keras 解决加载lstm+crf模型出错的问题
2020/06/10 Python
阿提哈德航空官方网站:Etihad Airways
2017/01/06 全球购物
园林施工员岗位职责
2013/12/11 职场文书
个人求职简历中英文自我评价
2013/12/16 职场文书
职业生涯规划书的格式
2013/12/29 职场文书
报社实习生自荐信
2014/01/24 职场文书
服装设计专业求职信
2014/06/16 职场文书
元旦晚会活动总结
2014/07/09 职场文书
2014年小学重阳节活动策划方案
2014/09/16 职场文书
保险公司增员口号
2015/12/25 职场文书
用 Python 元类的特性实现 ORM 框架
2021/05/19 Python
Java8中Stream的一些神操作
2021/11/02 Java/Android