目录

  • 1 里氏替换原则
  • 2 里氏替换原则应用

1 里氏替换原则

里氏替换原则(Liskov Substitution Principle,LSP)是指如果对每一个类型为 T1 的对象 o1,都有类型为 T2 的对象 o2,使得以 T1 定义的所有程序 P 在所有的对象 o1 都替换成 o2 时,程序 P 的行为没有发生变化,那么类型 T2 是类型 T1 的子类型。

定义看上去还是比较抽象,我们重新理解一下,可以理解为一个软件实体如果适用一个父类的话,那一定是适用于其子类,所有引用父类的地方必须能透明地使用其子类的对象,子类对象能够替换父类对象,而程序逻辑不变。根据这个理解,我们总结一下:

引申含义:子类可以扩展父类的功能,但不能改变父类原有的功能。
1、子类可以实现父类的抽象方法,但不能覆盖父类的非抽象方法
2、子类中可以增加自己特有的方法。
3、当子类的方法重载父类的方法时,方法的前置条件(即方法的输入/入参)要比父类方法的输入参数更宽松。
4、当子类的方法实现父类的方法时(重写/重载或实现抽象方法),方法的后置条件(即方法的输出/返回值)要比父类更严格或相等。

在前面讲开闭原则的时候埋下了一个伏笔,我们记得在获取折后时重写覆盖了父类的 getPrice()方法,增加了一个获取源码的方法 getOriginPrice(),显然就违背了里氏替换原则。我们修改一下代码,不应该覆盖 getPrice()方法,增加 getDiscountPrice()方法:
之前:

    public class JavaDiscountCourse extends JavaCourse {public JavaDiscountCourse(Integer id, String name, Double price) {super(id, name, price);}public Double getOriginPrice(){return super.getPrice();}public Double getPrice(){return super.getPrice() * 0.61;}}

更正:

public class JavaDiscountCourse extends JavaCourse {public JavaDiscountCourse(Integer id, String name, Double price) {super(id, name, price);}public Double getDiscountPrice(){return super.getPrice() * 0.61;}
}

2 里氏替换原则应用

使用里氏替换原则有以下优点:

1、约束继承泛滥,开闭原则的一种体现。
2、加强程序的健壮性,同时变更时也可以做到非常好的兼容性,提高程序的维护性、扩展性。降低需求变更时引入的风险。现在来描述一个经典的业务场景,用正方形、矩形和四边形的关系说明里氏替换原则,我们都知道正方形是一个特殊的长方形,那么就可以创建一个长方形父类 Rectangle 类:

public class Rectangle {private long height;private long width;public long getHeight() {return height;}public void setHeight(long height) {this.height = height;}public long getWidth() {return width;}public void setWidth(long width) {this.width = width;}
}

创建正方形 Square 类继承长方形:

public class Square extends Rectangle {private long length;public long getLength() {return length;}public void setLength(long length) {this.length = length;}@Overridepublic long getHeight() {return getLength();}@Overridepublic void setHeight(long height) {setLength(height);}@Overridepublic long getWidth() {return getLength();}@Overridepublic void setWidth(long width) {setLength(width);}
}

在测试类中创建 resize()方法,根据逻辑长方形的宽应该大于等于高,我们让高一直自增,知道高等于宽变成正方形:

public static void resize(Rectangle rectangle){while (rectangle.getWidth() >= rectangle.getHeight()){rectangle.setHeight(rectangle.getHeight() + 1);System.out.println("Width:" +rectangle.getWidth() +",Height:" + rectangle.getHeight());}System.out.println("Resize End,Width:" +rectangle.getWidth() +",Height:" + rectangle.getHeight());
}

测试代码:

public static void main(String[] args) {Rectangle rectangle = new Rectangle();
rectangle.setWidth(20);
rectangle.setHeight(10);
resize(rectangle);
}

运行结果:

发现高比宽还大了,在长方形中是一种非常正常的情况。现在我们再来看下面的代码,把长方形Rectangle 替换成它的子类正方形 Square,修改测试代码:

public static void main(String[] args) {Square square = new Square();square.setLength(10);resize(square);
}

这时候我们运行的时候就出现了死循环,违背了里氏替换原则,将父类替换为子类后,程序运行结果没有达到预期。因此,我们的代码设计是存在一定风险的。里氏替换原则只存在父类与子类之间,约束继承泛滥。我们再来创建一个基于长方形与正方形共同的抽象四边形 Quadrangle 接口:

public interface QuadRangle {long getWidth();long getHeight();
}

修改长方形 Rectangle 类:

public class Rectangle implements QuadRangle {private long height;
private long width;public long getHeight() {return height;}public void setHeight(long height) {this.height = height;}public long getWidth() {return width;}public void setWidth(long width) {this.width = width;}
}

修改正方形类 Square 类:

public class Square implements QuadRangle {private long length;public long getLength() {return length;}public void setLength(long length) {this.length = length;}public long getWidth() {return length;}public long getHeight() {return length;}
}

此时,如果我们把 resize()方法的参数换成四边形 Quadrangle 类,方法内部就会报错。因为正方形 Square 已经没有了 setWidth()和 setHeight()方法了。因此,为了约束继承泛滥,resize()的方法参数只能用 Rectangle 长方形。当然,我们在后面的设计模式中还会继续深入讲解。

七大设计原则之里氏替换原则应用相关推荐

  1. 软件设计原则之里氏替换原则、依赖倒置原则

    系列文章目录 软件设计原则之单一职责原则.开闭原则 软件设计原则之里氏替换原则.依赖倒置原则 软件设计原则之接口隔离原则.合成复用原则.迪米特原则 文章目录 系列文章目录 一.里氏替换原则 什么是里氏 ...

  2. 6大设计原则之里氏替换原则

    面对对象中的继承 优点如下: 代码共享,减少创建类的工作量,每个子类都拥有父类的方法和属性 提高代码的重用性 子类可以形如父类,但又异于父类 提高代码的可扩展性,很多开源框架的扩展接口都是通过继承父类 ...

  3. 开闭原则与里氏替换原则

    1.开闭原则 是面向对象设计的基本原则之一,是"可复用设计"的基础,它的主要原则是:对扩展开放,对修改关闭:意思就是我们改变一个软件时.应该通过扩展方式来改变软件,而不是修改原有的 ...

  4. python里氏替换原则_设计模式六大原则之里氏替换原则

    这是设计模式6 大原则系列的第二篇文章,附上前一篇文章地址:设计模式六大原则之单一职责原则.本文主要讲解设计模式的里氏替换原则. 肯定有不少人跟我刚看到这项原则的时候一样,对这个原则的名字充满疑惑.其 ...

  5. 深入理解开闭原则、里氏替换原则

    开闭原则(Open-Closed Principle)里氏替换原则 开闭原则(Open-Closed Principle) What 什么是开闭原则? Why 为什么要使用开闭原则和When 什么时候 ...

  6. 面向对象七大设计原则之里氏替换原则

    熟练掌握和应用面向对象设计七大原则,是初中级工程师向高级/资深工程师进阶的一个必备技能.他可以大大的提升程序的可复用性和可维护性是重构代码的一大利器. 本人菜鸟一枚本面文章的出发点是1.加深个人印象. ...

  7. java里氏替换原则例子_java 设计原则(六)里氏替换原则

    定义:如果对每一个类型T1的对象o1,都有类型为T2的对象o2,使得以T1定义的所有程序P在所有的对象o1都替换成o2时,程序P的行为没有发生变化,那么类型T2是类型T1的子类型. 定义扩展:一个软件 ...

  8. 里氏替换原则_春辉带你了解面相对象设计第二原则(里氏替换原则)

    里氏替换原则的定义 里氏替换原则(Liskov Substitution Principle,LSP)由麻省理工学院计算机科学实验室的里斯科夫(Liskov)女士在 1987 年的"面向对象 ...

  9. java solid设计原则_六大设计原则之里氏替换原则(LSP)

    一.SOLID 设计模式的六大原则有: Single Responsibility Principle:单一职责原则 Open Closed Principle:开闭原则 Liskov Substit ...

最新文章

  1. C# ConfigurationManager不存在问题解决
  2. 浏览器插件 火狐插件
  3. 百度语音识别技术负责人李先刚:如何利用Deep CNN大幅提升识别准确率?
  4. 用C#写的一个注册表操作类
  5. 浅谈PHP自动化代码审计技术
  6. 完成AOP 顶层设计-AopProxy
  7. SQL Server 2008使用LINQ进行数据访问(转载自IT168 [ http://www.it168.com/ ])
  8. Ajax4Jsf 简单介绍
  9. 释放变量所指向的内存_前端基础突破(二)内存回收与内存泄漏
  10. C++之继承探究(九):多态的代价
  11. 网络之美:JavaScript中Get和Set访问器的实现
  12. 程序员面试宝典(一) - 流程概览
  13. Linux杂谈之unzip命令
  14. 移动机器人路径规划算法综述(二)
  15. 如何通过蒲公英批量获取iPhone设备的udid
  16. gitbub图片 头像 图标不显示
  17. 360中不显示html中图片不显示图片,360极速浏览器无法显示图片解决方法详解
  18. Endless Dice 游戏解析
  19. 我是如何一步一步爬上 「64K限制」 的坑
  20. WAVE SUMMIT+峰会,飞桨八大亮点发布,全面助力产业智能化

热门文章

  1. 基于注意力机制的循环神经网络对 金融时间序列的应用 学习记录
  2. 如何建一个STM32F030工程模板(标准库版)
  3. 公安三维电子沙盘 警用数字沙盘系统
  4. AES加密:PHP与Java互通,解密准确
  5. 三六零发布半年报,一点两翼何时展翅高飞?
  6. 极域卸载、忘记密码及找回密码----骚操作
  7. 使用割圆术计算圆周率
  8. openpyxl复制图片问题
  9. 关于速卖通你应该知道的一些操作
  10. SparkSql On Hive