PHP面向对象五大原则之里氏替换原则(LSP)详解


Posted in PHP onApril 08, 2018

本文实例讲述了PHP面向对象五大原则之里氏替换原则(LSP)。分享给大家供大家参考,具体如下:

替换原则由MIT计算机科学实验室的Liskov女士在1987年的OOPSLA大会上的一篇文章中提出,主要阐述有关继承的一些原则,故称里氏替换原则。

2002年,Robert C.Martin出版了一本名为《Agile Software Development Principles Patterns and Practices》的书,在书中他把里氏替换原则最终简化成一句话:“Subtypes must be substitutable for their base types”(子类必须能够替换成它们的基类。)

1. LSP的内容

里氏替换原则(Liskov Substitution Principle, LSP)的定义和主要思想如下:由于面向对象编程技术中的继承在具体的编程中过于简单,在许多系统的设计和编程实现中,我们并没有认真地、理性地思考应用系统中各个类之间的继承关系是否合适,派生类是否能正确地对其基类中的某些方法进行重写等问题。因此经常出现滥用继承或者错误地进行了继承等现象,给系统的后期维护带来了不少麻烦。这就需要我们有一个设计原则来遵循,它就是替换原则。

LSP指出:子类类型必须能够替换掉它们的父类型、并出现在父类能够出现的任何地方。它指导我们如何正确地进行继承和派生,并合理地重用代码。此原则认为,一个软件实体如果使用一个基类的话,那么一定适用于其子类,而且这根本不能察觉出基类对象和子类对象的区别。想一想,是不是和多态的概念比较像?

2. LSP主要是针对继承的设计原则

因为继承与派生是OOP的一个主要特性,能够减少代码的重复编程实现,从而实现系统中的代码复用,但是如何正确地进行继承设计和合理地应用继承机制呢?

这就是LSP所要解决的问题:

如何正确地进行继承方面的设计?

最佳的继承层次如何获得?

怎么样避免所设计的类层次陷入不符合OCP原则的状况?

那如何遵守该设计原则呢?

1)父类的方法都要在子类中实现或者重写,并且派生类只实现其抽象类中声明的方法,而不应当给出多余的方法定义或实现

2)在客户端程序中只应该使用父类对象而不应当直接使用子类对象,这样可以实现运行期绑定(动态多态)。

如果A、B两个类违反了LSP的设计,通常的做法是创建一个新的抽象类C,作为两个具体类的超类,奖A和B的共同行为移动到C中,从而解决A和B的行为不完全一致的问题。

不过PHP对LSP的支持并不好,缺乏向上转型等概念,只能通过一些曲折的方法实现。对于这个原则,这里就不细讲了。

下面给出一个缓存的实现接口,用抽象类做基类,遵循LSP实现其设计。

<?php
abstract class Cache
{
 /**
  * 设置一个缓存变量
  * @param $key 缓存key
  * @param $value 缓存内容
  * @param int $expire 缓存时间(秒)
  * @return boolean 是否缓存成功
  */
 public abstract function set($key, $value, $expire = 60);
 /**
  * 获取一个已经缓存的
  * @param $key 缓存key
  * @return mixed 缓存内容
  */
 public abstract function get($key);
 /**
  * 删除一个已经缓存的变量
  * @param $key 缓存key
  * @return boolean 是否删除成功
  */
 public abstract function del($key);
 /**
  * 删除全部缓存变量
  * @return boolean 是否删除成功
  */
 public abstract function delAll();
 /**
  * 检测是否存在对应的缓存
  * @param $key 缓存key
  * @return boolean 是否存在
  */
 public abstract function has($key);
}

如果现在要求实现文件、memcache、accelerator等各种机制下的缓存,只需要继承这个抽象类并实现其抽象方法即可。

LSP中代码的不仅仅是功能,还名手语意。试思考:白马可以代换马,而牛同样作为劳动力,可代换马否?高跟鞋也是鞋子,男人穿高跟鞋又是否能接受?

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

PHP 相关文章推荐
php 动态添加记录
Mar 10 PHP
PHP 实现多服务器共享 SESSION 数据
Aug 15 PHP
phpMyAdmin出现无法载入 mcrypt 扩展,请检查PHP配置的解决方法
Mar 26 PHP
Codeigniter注册登录代码示例
Jun 12 PHP
ThinkPHP之M方法实例详解
Jun 20 PHP
PHP解密Unicode及Escape加密字符串
May 17 PHP
Zend Framework教程之配置文件application.ini解析
Mar 10 PHP
php 从一个数组中随机的取出若干个不同的数实例
Dec 31 PHP
Laravel中获取路由参数Route Parameters的五种方法示例
Sep 29 PHP
laravel5 Eloquent 实现事务方式
Oct 21 PHP
yii框架结合charjs实现统计30天数据的方法
Apr 04 PHP
为你的 Laravel 验证器加上多验证场景的实现
Apr 07 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
PHP设计模式之原型模式定义与用法详解
Apr 03 #PHP
You might like
php中$this-&amp;gt;含义分析
2009/11/29 PHP
PHP生成唯一订单号
2015/07/05 PHP
PHP使用xpath解析XML的方法详解
2017/05/20 PHP
JavaScript 加号(+)运算符号
2009/12/06 Javascript
jquery一般方法介绍 入门参考
2011/06/21 Javascript
editable.js 基于jquery的表格的编辑插件
2011/10/24 Javascript
JS获取并操作iframe中元素的方法
2013/03/21 Javascript
javascript实现在网页中运行本地程序的方法
2016/02/03 Javascript
JS实现弹出下载对话框及常见文件类型的下载
2017/07/13 Javascript
js仿微信抢红包功能
2020/09/25 Javascript
通过js动态创建标签,并设置属性方法
2018/02/24 Javascript
解决vue.js 数据渲染成功仍报错的问题
2018/08/25 Javascript
微信小程序自定义可滑动日历界面
2018/12/28 Javascript
javascript实现点击星星小游戏
2019/12/24 Javascript
Vue-router编程式导航的两种实现代码
2021/03/04 Vue.js
[01:25:33]完美世界DOTA2联赛PWL S3 INK ICE vs Magma 第二场 12.20
2020/12/23 DOTA
Python获取电脑硬件信息及状态的实现方法
2014/08/29 Python
python实现的守护进程(Daemon)用法实例
2015/06/02 Python
Python基于opencv调用摄像头获取个人图片的实现方法
2019/02/21 Python
在macOS上搭建python环境的实现方法
2019/08/13 Python
利用Python裁切tiff图像且读取tiff,shp文件的实例
2020/03/10 Python
浅谈keras中的Merge层(实现层的相加、相减、相乘实例)
2020/05/23 Python
快速一键生成Python爬虫请求头
2021/03/04 Python
html5写一个BUI折叠菜单插件的实现方法
2019/09/11 HTML / CSS
美国最大的网上冲印店:Shutterfly
2017/01/01 全球购物
Boden澳大利亚官网:英国在线服装公司
2018/08/05 全球购物
Chain Reaction Cycles俄罗斯:世界上最大的在线自行车商店
2019/08/27 全球购物
德国前卫设计师时装在线商店:Luxury Loft
2019/11/04 全球购物
应届生找工作求职信
2014/06/24 职场文书
商场周年庆活动方案
2014/08/19 职场文书
收入及婚姻状况证明
2014/11/20 职场文书
教导主任个人总结
2015/03/03 职场文书
新手初学Java网络编程
2021/07/07 Java/Android
MySQL中B树索引和B+树索引的区别详解
2022/03/03 MySQL
MySql分区类型及创建分区的方法
2022/04/13 MySQL
tomcat默认最大连接数及相关调整方法
2022/05/06 Servers