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

里氏替换原则:Java面向对象设计的基石

        在面向对象编程(OOP)中,继承是一个强大的工具,它允许我们创建新的类(子类)来复用和扩展现有类(父类)的功能。然而,继承也带来了复杂性,特别是在确保子类能够正确替换父类而不破坏程序行为方面。为了解决这个问题,里氏替换原则(Liskov Substitution Principle,LSP)应运而生。本文将详细介绍里氏替换原则的概念、重要性、实践方法,并通过Java代码示例来加深理解。

 

一、里氏替换原则的概念

里氏替换原则由麻省理工学院的芭芭拉·利斯科夫(Barbara Liskov)在1987年提出。其核心思想是:在软件系统中,子类对象应该能够替换父类对象,并且替换后程序的行为应该保持不变。换句话说,如果父类对象可以在某个地方被使用,那么子类对象也应该能够无障碍地替换它,而不会改变程序的行为。

里氏替换原则的定义可以表述为:如果对每一个类型为T1的对象p,都有类型为T2的对象q,使得以T1定义的所有程序在应用于p时,能够以相同的结果运行在q上,那么类型T2的对象q就可以替换类型T1的对象p。

这个原则确保了继承的正确性和软件的可扩展性,是面向对象设计(OOD)和面向对象程序设计(OOP)中的一个基本原则。

二、里氏替换原则的重要性

里氏替换原则的重要性体现在以下几个方面:

  1. 增强程序的健壮性:通过确保子类能够正确替换父类,里氏替换原则降低了需求变更时引入的风险,提高了程序的稳定性和可靠性。

  2. 提高代码的可维护性:遵循里氏替换原则,可以使得代码更加清晰、易于理解和维护。当需要修改或扩展功能时,可以通过添加新的子类来实现,而不需要修改现有的父类代码。

  3. 增强代码的可扩展性:里氏替换原则鼓励使用抽象类和接口来定义基类,这样可以在运行时确定具体的实现方式,增加了系统的灵活性。

  4. 降低耦合性:通过遵循里氏替换原则,可以减少子类对父类的依赖,从而降低代码的耦合性,使得系统更加易于修改和扩展。

三、里氏替换原则的实践方法

要实践里氏替换原则,需要遵循以下几个关键步骤:

  1. 子类必须完全实现父类的方法:子类应该能够正确实现父类的所有方法,包括抽象方法和非抽象方法。如果子类不能完整实现父类的方法,或者父类的某些方法在子类中已经发生“畸变”,那么建议断开父子继承关系,采用依赖、聚集、组合等关系代替继承。

  2. 子类可以有自己的个性:虽然子类需要完全实现父类的方法,但子类也可以添加自己的方法和属性,以扩展功能。这些新增的方法和属性不应该影响父类已经定义的行为。

  3. 覆盖或实现父类的方法时,输入参数可以被放大:当子类覆盖或实现父类的方法时,输入参数的类型可以比父类方法中的参数类型更宽松(即范围更大)。这样做可以使得子类能够处理更多的输入情况,而不会破坏父类方法的行为。

  4. 覆盖或实现父类的方法时,输出结果可以被缩小:当子类覆盖或实现父类的方法时,输出结果的类型可以比父类方法中的返回类型更具体(即范围更小)。这样做可以确保子类方法返回的结果更加精确,同时也不会破坏父类方法的行为。

四、Java代码示例

        下面通过几个Java代码示例来进一步说明里氏替换原则的实践方法。

示例1:鸟类和企鹅类的关系

        在这个示例中,我们定义了一个鸟类(Bird)作为基类,并定义了一个企鹅类(Penguin)作为鸟类的子类。然而,企鹅虽然属于鸟类,但它不会飞。因此,如果我们在鸟类中定义了一个飞行方法(fly),并在企鹅类中重写了这个方法(将其设置为不飞行),那么就会违反里氏替换原则。

public class Bird {public double flySpeed;public void setFlySpeed(double speed) {this.flySpeed = speed;}public double getTimeToFly(double distance) {return distance / flySpeed;}
}public class Penguin extends Bird {@Overridepublic void setFlySpeed(double speed) {this.flySpeed = 0; // 企鹅不会飞,飞行速度设置为0}
}public class Test {public static void main(String[] args) {Bird bird = new Penguin();bird.setFlySpeed(110);try {System.out.println("企鹅飞了" + bird.getTimeToFly(200) + "公里");} catch (Exception e) {System.out.println("出现错误");}}
}

        在这个示例中,由于企鹅类重写了鸟类的飞行方法,导致当使用企鹅对象替换鸟类对象时,程序的行为发生了变化(出现了除以零的错误)。因此,这个设计违反了里氏替换原则。

        为了解决这个问题,我们可以将鸟类和企鹅类的关系重新设计。我们可以定义一个更一般的基类(如动物类),并让鸟类和企鹅类都继承自这个基类。这样,企鹅类就可以拥有自己特有的行为(如游泳),而不会破坏鸟类已经定义的行为(如飞行)。

示例2:形状类和矩形类的关系

        在这个示例中,我们定义了一个形状类(Shape)作为基类,并定义了一个矩形类(Rectangle)作为形状类的子类。同时,我们还定义了一个正方形类(Square),它也可以看作是形状类的一个子类(尽管在几何学中正方形是特殊的长方形,但在这个示例中我们将其视为独立的类)。

public class Shape {public virtual int GetArea() {return 0;}
}public class Rectangle : Shape {public int Width { get; set; }public int Height { get; set; }public override int GetArea() {return Width * Height;}
}public class Square : Shape {public int SideLength { get; set; }public override int GetArea() {return SideLength * SideLength;}
}public class Program {public static void Main(string[] args) {Shape rectangle = new Rectangle { Width = 5, Height = 4 };Console.WriteLine("Rectangle Area: " + rectangle.GetArea()); // 输出 20Shape square = new Square { SideLength = 5 };Console.WriteLine("Square Area: " + square.GetArea()); // 输出 25}
}

        在这个示例中,矩形类和正方形类都继承自形状类,并且各自实现了自己的GetArea方法。由于这两个类都正确地实现了形状类的方法,并且没有增加父类不具备的行为,因此它们符合里氏替换原则。

总结

        里氏替换原则是面向对象设计中的一个重要原则,它确保了子类能够正确替换父类而不破坏程序的行为。通过遵循里氏替换原则,我们可以增强程序的健壮性、可维护性和可扩展性,同时降低需求变更时引入的风险。

        在实践中,我们需要确保子类完全实现父类的方法,并且不增加父类不具备的行为。同时,我们还需要注意子类方法的前置条件和后置条件,以确保它们与父类方法保持一致。

相关文章:

里氏替换原则:Java面向对象设计的基石

在面向对象编程(OOP)中,继承是一个强大的工具,它允许我们创建新的类(子类)来复用和扩展现有类(父类)的功能。然而,继承也带来了复杂性,特别是在确保子类能够正…...

恒创科技:服务器操作系统和客户端操作系统之间的区别

客户端操作系统和服务器操作系统是两种不同的操作系统,旨在满足计算机网络环境中的特定目的。虽然每种类型的操作系统在基本功能方面都有一些相似之处,但它们针对不同的用例进行了优化,并具有针对其特定角色量身定制的特定功能。 什么是服务器…...

做异端中的异端 -- Emacs裸奔之路4: 你不需要IDE

确切地说,你不需要在IDE里面编写或者阅读代码。 IDE用于Render资源文件比较合适,但处理文本,并不划算。 这的文本文件,包括源代码,配置文件,文档等非二进制文件。 先说说IDE带的便利: 函数或者变量的自动…...

Unity3d C# 摄像头检测敌方单位(目标层级)并在画面中标注(含源码)

前言 需要实现的功能是通过一个专门的检测摄像头将出现在摄像头画面内的敌方单位检测出来,并通过框选的UI框在画面中标记出来。检测摄像头支持自动检测和手动控制检测,同时需要实现锁定模式,检测到一个敌方单位直接锁定到对象上等功能。 效…...

js 16进制加密

function hexEncode(str) { let hexEncodedStr ‘’; for (let i 0; i < str.length; i) { let charCode str.charCodeAt(i); let hexCode charCode.toString(16).padStart(2, ‘0’); hexEncodedStr ‘\x’ hexCode; } return hexEncodedStr; } // 示例用法 let ori…...

性能测试之压测

1、首先需要提前准备好需要压测的接口地址及对应的接口参数 写好对应的压测接口及对应参数脚本 2、添加线程组&#xff08;根据对应的需求提供的QPS及需要压测的数量如有&#xff09; 如&#xff1a;40个线程&#xff0c;循环次数为永远&#xff08;或者根据自身情况设置循…...

CentOS修改yum.repos.d源,避免“Could not resolve host: mirrorlist.centos.org”错误

1、问题现象 由于CentOS停止维护&#xff0c;mirrorlist.centos.org网站也关闭不可访问。导致CentOS默认配置的yum.repos.d源也不可用&#xff0c;所以执行yum命令会报“Could not resolve host: mirrorlist.centos.org”错误。具体如下&#xff1a; Could not retrieve mirror…...

Python 三目运算实战详解

Python 的三目运算符&#xff08;也称为条件表达式&#xff09;是一种简洁的方式来执行基于条件的赋值或返回值。它的语法类似于其他编程语言中的三元运算符&#xff0c;但有一些细微的不同。在 Python 中&#xff0c;三目运算符的语法如下&#xff1a; value_if_true if cond…...

JVM 性能调优 -- CMS 垃圾回收器 GC 日志分析【Full GC】

前言&#xff1a; 上一篇我们分析了 Minor GC 的发生过程&#xff0c;因为 GC 日志没有按我们预估的思路进行打印&#xff0c;其中打印了 CMS 垃圾回收器的部分日志&#xff0c;本篇我们就来分析一下 CMS 垃圾收集日志。 JVM 系列文章传送门 初识 JVM&#xff08;Java 虚拟机…...

PS的学习

背景差色较大&#xff0c;就魔棒 魔棒的连续就是倒水点的跨越问题 魔棒的容差的选择就有点看经验了&#xff0c;看颜色的统一程度选择 Ctrl D 取消当前所有的选区 至于快速选择工具&#xff0c;和对象选择工具也差不多&#xff0c;只不过控制范围变成了一块一块的&#x…...

数据集搜集器(百科)008

对数据集搜集器&#xff08;百科&#xff09;007进行一下改进&#xff1a; 错误处理&#xff1a;增加更多的错误处理&#xff0c;比如网络请求超时、解析错误等。 用户界面&#xff1a;增加一些提示信息&#xff0c;让用户更清楚当前的操作状态。 多线程处理&#xff1a;确保多…...

Java学习,反射

Java反射是Java编程语言的一个重要特性&#xff0c;它允许程序在运行时查看任意对象所属的类&#xff0c;获取类的内部信息&#xff08;包括构造器、字段和方法等&#xff09;&#xff0c;并能动态地调用对象的方法或构造器。 反射概念 反射&#xff08;Reflection&#xff09…...

数据结构 (18)数的定义与基本术语

前言 数据结构是计算机科学中的一个核心概念&#xff0c;它描述了数据元素之间的关系以及这些元素在计算机中的存储方式。 一、数的定义 在计算机科学中&#xff0c;“数”通常指的是树形数据结构&#xff0c;它是一种非线性的数据结构&#xff0c;由节点&#xff08;或称为元素…...

Flink的双流join理解

如何保证Flink双流Join准确性和及时性、除了窗口join还存在哪些实现方式、究竟如何回答才能完全打动面试官呢。。你将在文中找到答案。 1 引子 1.1 数据库SQL中的JOIN 我们先来看看数据库SQL中的JOIN操作。如下所示的订单查询SQL&#xff0c;通过将订单表的id和订单详情表ord…...

《使用Python进行数据挖掘:理论、应用与案例研究》

嘿&#xff0c;今天我要给你们介绍一本使用Python进行数据挖掘的好书。这本书是由吴迪博士撰写的&#xff0c;他是雷曼学院商学院的助理教授&#xff0c;也是数据科学的实战派。 在这个时代&#xff0c;数据多得让人眼花缭乱&#xff0c;要从中找出有用的信息&#xff0c;那可不…...

Go语言技巧:快速统一字符串中的换行符,解决跨平台问题

统一字符串中的 Windows \r\n 换行符 — Go语言实现 在编程中&#xff0c;尤其是处理跨平台的文本数据时&#xff0c;换行符的处理是一个常见的问题。Windows 系统使用 \r\n 作为换行符&#xff0c;而 Unix-like 系统&#xff08;如 Linux 和 macOS&#xff09;使用 \n。在 Go…...

算法训练营day20(二叉树06:最大二叉树,合并二叉树,搜索二叉树,验证搜索二叉树)

第六章 二叉树 part06 今日内容 ● 654.最大二叉树 ● 617.合并二叉树 ● 700.二叉搜索树中的搜索 ● 98.验证二叉搜索树 详细布置 654.最大二叉树 又是构造二叉树&#xff0c;昨天大家刚刚做完 中序后序确定二叉树&#xff0c;今天做这个 应该会容易一些&#xff0c; 先看视…...

Leetcode(区间合并习题思路总结,持续更新。。。)

讲解题目&#xff1a;合并区间 以数组 intervals 表示若干个区间的集合&#xff0c;其中单个区间为 intervals[i] [starti, endi] 。请你合并所有重叠的区间&#xff0c; 并返回一个不重叠的区间数组&#xff0c;该数组需恰好覆盖输入中的所有区间。示例 1&#xff1a;输入&a…...

『python爬虫』使用docling 将pdf或html网页转为MD (保姆级图文)

目录 预览效果安装下载模型测试代码总结 欢迎关注 『python爬虫』 专栏&#xff0c;持续更新中 欢迎关注 『python爬虫』 专栏&#xff0c;持续更新中 预览效果 支持转化pdf的表格 安装 Docling 本身是专注于文档转换的工具&#xff0c;通常用于将文件&#xff08;如 PDF&…...

elasticsearch现有集群扩展节点

原文地址&#xff1a;elasticsearch现有集群扩展节点 – 无敌牛 欢迎参观我的个人博客&#xff1a;无敌牛 – 技术/著作/典籍/分享等 给现有的 elasticsearch 集群扩展节点比较容易&#xff0c;已有的集群不需要做任何修改&#xff0c;也不用对服务做任何处理&#xff0c;只需…...

应用升级/灾备测试时使用guarantee 闪回点迅速回退

1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间&#xff0c; 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点&#xff0c;不需要开启数据库闪回。…...

工业安全零事故的智能守护者:一体化AI智能安防平台

前言&#xff1a; 通过AI视觉技术&#xff0c;为船厂提供全面的安全监控解决方案&#xff0c;涵盖交通违规检测、起重机轨道安全、非法入侵检测、盗窃防范、安全规范执行监控等多个方面&#xff0c;能够实现对应负责人反馈机制&#xff0c;并最终实现数据的统计报表。提升船厂…...

23-Oracle 23 ai 区块链表(Blockchain Table)

小伙伴有没有在金融强合规的领域中遇见&#xff0c;必须要保持数据不可变&#xff0c;管理员都无法修改和留痕的要求。比如医疗的电子病历中&#xff0c;影像检查检验结果不可篡改行的&#xff0c;药品追溯过程中数据只可插入无法删除的特性需求&#xff1b;登录日志、修改日志…...

ssc377d修改flash分区大小

1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...

Axios请求超时重发机制

Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式&#xff1a; 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...

前端开发面试题总结-JavaScript篇(一)

文章目录 JavaScript高频问答一、作用域与闭包1.什么是闭包&#xff08;Closure&#xff09;&#xff1f;闭包有什么应用场景和潜在问题&#xff1f;2.解释 JavaScript 的作用域链&#xff08;Scope Chain&#xff09; 二、原型与继承3.原型链是什么&#xff1f;如何实现继承&a…...

重启Eureka集群中的节点,对已经注册的服务有什么影响

先看答案&#xff0c;如果正确地操作&#xff0c;重启Eureka集群中的节点&#xff0c;对已经注册的服务影响非常小&#xff0c;甚至可以做到无感知。 但如果操作不当&#xff0c;可能会引发短暂的服务发现问题。 下面我们从Eureka的核心工作原理来详细分析这个问题。 Eureka的…...

以光量子为例,详解量子获取方式

光量子技术获取量子比特可在室温下进行。该方式有望通过与名为硅光子学&#xff08;silicon photonics&#xff09;的光波导&#xff08;optical waveguide&#xff09;芯片制造技术和光纤等光通信技术相结合来实现量子计算机。量子力学中&#xff0c;光既是波又是粒子。光子本…...

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数 在软件开发中,单例模式(Singleton Pattern)是一种常见的设计模式,确保一个类仅有一个实例,并提供一个全局访问点。在多线程环境下,实现单例模式时需要注意线程安全问题,以防止多个线程同时创建实例,导致…...

2025年渗透测试面试题总结-腾讯[实习]科恩实验室-安全工程师(题目+回答)

安全领域各种资源&#xff0c;学习文档&#xff0c;以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具&#xff0c;欢迎关注。 目录 腾讯[实习]科恩实验室-安全工程师 一、网络与协议 1. TCP三次握手 2. SYN扫描原理 3. HTTPS证书机制 二…...