当前位置: 首页 > news >正文

软件设计原则-里氏替换原则讲解以及代码示例

里氏替换原则

一,介绍

1.前言

里氏替换原则(Liskov Substitution Principle,LSP)是面向对象设计中的一条重要原则,它由Barbara Liskov在1987年提出。

里氏替换原则的核心思想是:父类的对象可以被子类的对象替换,而程序的行为不会发生变化。也就是说,如果一个类型A是另一个类型B的子类型,那么在任何使用B的地方都可以使用A,而不会引起错误或异常。

2.何时使用里氏替换原则

  1. 当需要编写基类或抽象类时:在编写基类或抽象类时应该尽可能地遵循里氏替换原则,以保证后续的子类能够正确地继承和使用基类的接口或者抽象类的方法。

  2. 当需要对已有的代码进行重构时:在重构已有的代码时,我们可以通过遵循里氏替换原则,使得代码更加易于理解、扩展和维护。通过将某些动态绑定的行为转化为静态绑定的行为,可以降低代码的复杂度并增强其可控性。

  3. 当需要进行单元测试或集成测试时:在进行单元测试或集成测试时,我们可以使用子类对象来替换父类对象,以确保测试结果的准确性。如果使用子类对象无法替换相应的父类对象,则表示可能存在设计上的问题,需要进一步优化。

  4. 当需要扩展系统的功能时:在扩展系统的功能时,我们应该尽可能地遵循里氏替换原则,以确保新的组件能够与现有的组件正常协作。通过使用基类或抽象类来定义接口,可以使得组件之间的耦合度更低。

二,代码示例

为了更详细地介绍里氏替换原则,我们可以通过一个例子来说明:

假设有一个图形计算程序,程序可以计算不同形状图形的面积。最初的设计可能会像这样:

class Shape {// 省略其他属性和方法public double calculateArea() {// 默认实现,返回0return 0;}
}class Rectangle extends Shape {private double width;private double height;// 省略构造方法和其他属性方法@Overridepublic double calculateArea() {return width * height;}
}class Circle extends Shape {private double radius;// 省略构造方法和其他属性方法@Overridepublic double calculateArea() {return Math.PI * radius * radius;}
}


这个设计看起来似乎没有问题,但问题在于当我们需要添加新的图形类型时,比如三角形,计算面积的方式与矩形和圆形不同,会导致父类的默认实现无法满足需求。

为了符合里氏替换原则,我们可以进行重构。首先,我们定义一个抽象类`Shape`:

abstract class Shape {public abstract double calculateArea();
}

然后,对每种具体的图形类型,创建一个子类并实现`calculateArea()`方法:

class Rectangle extends Shape {private double width;private double height;// 省略构造方法和其他属性方法@Overridepublic double calculateArea() {return width * height;}
}class Circle extends Shape {private double radius;// 省略构造方法和其他属性方法@Overridepublic double calculateArea() {return Math.PI * radius * radius;}
}class Triangle extends Shape {private double base;private double height;// 省略构造方法和其他属性方法@Overridepublic double calculateArea() {return 0.5 * base * height;}
}

现在,我们可以通过扩展子类来添加新的图形类型,而且每个子类都提供了自己的面积计算方式。

这个重构后的设计符合里氏替换原则,因为我们可以将子类的对象替换父类的对象,而不影响程序的行为。这样做的好处是,通过面向抽象编程,代码更加灵活、可扩展,同时也提高了系统的可维护性和可测试性。

总结起来,里氏替换原则强调了继承关系的正确使用,要求子类能够完全替代父类,而不破坏程序的正确性。遵循该原则可以提高代码的重用性、灵活性和可靠性,是良好的软件设计实践之一。

三,优缺点

优点:

  1. 提高代码的可复用性:遵循里氏替换原则可以确保子类对象能够替换父类对象,这意味着我们可以使用统一的接口或抽象类来处理一组对象,从而提高了代码的可复用性。

  2. 增强程序的可扩展性:通过良好的继承关系,可以在不修改现有代码的情况下,通过添加新的子类来扩展系统的功能。这样可以降低对原有代码的影响范围,提高了程序的可扩展性。

  3. 促进代码的层次化结构:通过定义好的抽象类或接口,可以将代码按照层次化的结构组织起来,提高代码的可读性和可维护性。

  4. 提高代码的可测试性:遵循里氏替换原则可以使得代码更易于进行单元测试,因为我们可以使用父类对象来代替子类对象进行测试,从而提高了代码的可测试性。

缺点:

  1. 过度约束:有时为了满足里氏替换原则,可能需要引入过多的抽象类或接口,导致代码变得复杂,增加了设计和开发的难度。

  2. 需要在继承关系上建立合适的层次结构:正确地使用里氏替换原则需要在继承关系上建立适当的层次结构,这需要设计者有较强的面向对象设计能力。

  3. 可能违反单一职责原则:为了满足里氏替换原则,有时需要在父类中定义多个不相关的接口或抽象方法,这可能违反了单一职责原则,导致代码的可读性和维护性下降。

总的来说,里氏替换原则通过良好的继承关系可以提高代码的可复用性、可扩展性和可测试性,但需要在继承关系的层次结构上做出合理的设计,并权衡与其他设计原则的关系。

http://www.lryc.cn/news/204674.html

相关文章:

  • Sui提供dApp Kit 助力快速构建React Apps和dApps
  • 2023年系统设计面试如何破解?进入 FAANG 面试的实战指南
  • (react+ts)vite项目中的路径别名的配置
  • 【MATLAB源码-第51期】基于matlab的粒子群算法(PSO)的栅格地图路径规划。
  • React之render
  • 基于springboot实现财务管理系统项目【项目源码+论文说明】计算机毕业设计
  • 设计模式:组合模式(C#、JAVA、JavaScript、C++、Python、Go、PHP)
  • 超强满血不收费的AI绘图教程来了(在线Stable Diffusion一键即用)
  • 【蓝桥每日一题]-动态规划 (保姆级教程 篇12)#照相排列
  • 纺织工厂数字孪生3D可视化管理平台,推动纺织产业数字化转型
  • 【七】SpringBoot为什么可以打成 jar包启动
  • 031-第三代软件开发-屏幕保护
  • Ubuntu 22.04 更新完内核重启卡在 grub 命令行解决办法
  • Stream流式处理
  • ROG STRIX GS-AX5400 使用笔记
  • 【刷题-PTA】堆栈模拟队列(代码+动态图解)
  • FileUpload控件上传文件时出现 不支持给定路径的格式.的解决方法
  • 这两天的一些碎碎念
  • Unity 最新DOTS系列之《Baking与Baker的详解》
  • 【Tensorflow 2.12 简单智能商城商品推荐系统搭建】
  • Unity 单例-接口模式
  • 【Java 进阶篇】Java XML解析:从入门到精通
  • 【图像配准】Canny边缘检测+模板配准红外可见光双路数据
  • 关于单机流程编排技术——docker compose安装使用的问题
  • Google Chrome的新“IP保护”功能将隐藏用户的IP地址
  • 做机器视觉工程师,苏州德创能不能去工作?
  • 交换机基础(二):VLAN 基础知识
  • 一个基于Vue3搭建的低代码数据可视化开发平台
  • 经验风险最小化与结构风险最小化:优化机器学习模型的两种方法
  • Java泛型中的问号是什么意思