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

一、里氏替换原则的概念
里氏替换原则由麻省理工学院的芭芭拉·利斯科夫(Barbara Liskov)在1987年提出。其核心思想是:在软件系统中,子类对象应该能够替换父类对象,并且替换后程序的行为应该保持不变。换句话说,如果父类对象可以在某个地方被使用,那么子类对象也应该能够无障碍地替换它,而不会改变程序的行为。
里氏替换原则的定义可以表述为:如果对每一个类型为T1的对象p,都有类型为T2的对象q,使得以T1定义的所有程序在应用于p时,能够以相同的结果运行在q上,那么类型T2的对象q就可以替换类型T1的对象p。
这个原则确保了继承的正确性和软件的可扩展性,是面向对象设计(OOD)和面向对象程序设计(OOP)中的一个基本原则。
二、里氏替换原则的重要性
里氏替换原则的重要性体现在以下几个方面:
-
增强程序的健壮性:通过确保子类能够正确替换父类,里氏替换原则降低了需求变更时引入的风险,提高了程序的稳定性和可靠性。
-
提高代码的可维护性:遵循里氏替换原则,可以使得代码更加清晰、易于理解和维护。当需要修改或扩展功能时,可以通过添加新的子类来实现,而不需要修改现有的父类代码。
-
增强代码的可扩展性:里氏替换原则鼓励使用抽象类和接口来定义基类,这样可以在运行时确定具体的实现方式,增加了系统的灵活性。
-
降低耦合性:通过遵循里氏替换原则,可以减少子类对父类的依赖,从而降低代码的耦合性,使得系统更加易于修改和扩展。
三、里氏替换原则的实践方法
要实践里氏替换原则,需要遵循以下几个关键步骤:
-
子类必须完全实现父类的方法:子类应该能够正确实现父类的所有方法,包括抽象方法和非抽象方法。如果子类不能完整实现父类的方法,或者父类的某些方法在子类中已经发生“畸变”,那么建议断开父子继承关系,采用依赖、聚集、组合等关系代替继承。
-
子类可以有自己的个性:虽然子类需要完全实现父类的方法,但子类也可以添加自己的方法和属性,以扩展功能。这些新增的方法和属性不应该影响父类已经定义的行为。
-
覆盖或实现父类的方法时,输入参数可以被放大:当子类覆盖或实现父类的方法时,输入参数的类型可以比父类方法中的参数类型更宽松(即范围更大)。这样做可以使得子类能够处理更多的输入情况,而不会破坏父类方法的行为。
-
覆盖或实现父类的方法时,输出结果可以被缩小:当子类覆盖或实现父类的方法时,输出结果的类型可以比父类方法中的返回类型更具体(即范围更小)。这样做可以确保子类方法返回的结果更加精确,同时也不会破坏父类方法的行为。
四、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、添加线程组(根据对应的需求提供的QPS及需要压测的数量如有) 如:40个线程,循环次数为永远(或者根据自身情况设置循…...
CentOS修改yum.repos.d源,避免“Could not resolve host: mirrorlist.centos.org”错误
1、问题现象 由于CentOS停止维护,mirrorlist.centos.org网站也关闭不可访问。导致CentOS默认配置的yum.repos.d源也不可用,所以执行yum命令会报“Could not resolve host: mirrorlist.centos.org”错误。具体如下: Could not retrieve mirror…...
Python 三目运算实战详解
Python 的三目运算符(也称为条件表达式)是一种简洁的方式来执行基于条件的赋值或返回值。它的语法类似于其他编程语言中的三元运算符,但有一些细微的不同。在 Python 中,三目运算符的语法如下: value_if_true if cond…...
JVM 性能调优 -- CMS 垃圾回收器 GC 日志分析【Full GC】
前言: 上一篇我们分析了 Minor GC 的发生过程,因为 GC 日志没有按我们预估的思路进行打印,其中打印了 CMS 垃圾回收器的部分日志,本篇我们就来分析一下 CMS 垃圾收集日志。 JVM 系列文章传送门 初识 JVM(Java 虚拟机…...
PS的学习
背景差色较大,就魔棒 魔棒的连续就是倒水点的跨越问题 魔棒的容差的选择就有点看经验了,看颜色的统一程度选择 Ctrl D 取消当前所有的选区 至于快速选择工具,和对象选择工具也差不多,只不过控制范围变成了一块一块的&#x…...
数据集搜集器(百科)008
对数据集搜集器(百科)007进行一下改进: 错误处理:增加更多的错误处理,比如网络请求超时、解析错误等。 用户界面:增加一些提示信息,让用户更清楚当前的操作状态。 多线程处理:确保多…...
Java学习,反射
Java反射是Java编程语言的一个重要特性,它允许程序在运行时查看任意对象所属的类,获取类的内部信息(包括构造器、字段和方法等),并能动态地调用对象的方法或构造器。 反射概念 反射(Reflection)…...
数据结构 (18)数的定义与基本术语
前言 数据结构是计算机科学中的一个核心概念,它描述了数据元素之间的关系以及这些元素在计算机中的存储方式。 一、数的定义 在计算机科学中,“数”通常指的是树形数据结构,它是一种非线性的数据结构,由节点(或称为元素…...
Flink的双流join理解
如何保证Flink双流Join准确性和及时性、除了窗口join还存在哪些实现方式、究竟如何回答才能完全打动面试官呢。。你将在文中找到答案。 1 引子 1.1 数据库SQL中的JOIN 我们先来看看数据库SQL中的JOIN操作。如下所示的订单查询SQL,通过将订单表的id和订单详情表ord…...
《使用Python进行数据挖掘:理论、应用与案例研究》
嘿,今天我要给你们介绍一本使用Python进行数据挖掘的好书。这本书是由吴迪博士撰写的,他是雷曼学院商学院的助理教授,也是数据科学的实战派。 在这个时代,数据多得让人眼花缭乱,要从中找出有用的信息,那可不…...
Go语言技巧:快速统一字符串中的换行符,解决跨平台问题
统一字符串中的 Windows \r\n 换行符 — Go语言实现 在编程中,尤其是处理跨平台的文本数据时,换行符的处理是一个常见的问题。Windows 系统使用 \r\n 作为换行符,而 Unix-like 系统(如 Linux 和 macOS)使用 \n。在 Go…...
算法训练营day20(二叉树06:最大二叉树,合并二叉树,搜索二叉树,验证搜索二叉树)
第六章 二叉树 part06 今日内容 ● 654.最大二叉树 ● 617.合并二叉树 ● 700.二叉搜索树中的搜索 ● 98.验证二叉搜索树 详细布置 654.最大二叉树 又是构造二叉树,昨天大家刚刚做完 中序后序确定二叉树,今天做这个 应该会容易一些, 先看视…...
Leetcode(区间合并习题思路总结,持续更新。。。)
讲解题目:合并区间 以数组 intervals 表示若干个区间的集合,其中单个区间为 intervals[i] [starti, endi] 。请你合并所有重叠的区间, 并返回一个不重叠的区间数组,该数组需恰好覆盖输入中的所有区间。示例 1:输入&a…...
『python爬虫』使用docling 将pdf或html网页转为MD (保姆级图文)
目录 预览效果安装下载模型测试代码总结 欢迎关注 『python爬虫』 专栏,持续更新中 欢迎关注 『python爬虫』 专栏,持续更新中 预览效果 支持转化pdf的表格 安装 Docling 本身是专注于文档转换的工具,通常用于将文件(如 PDF&…...
elasticsearch现有集群扩展节点
原文地址:elasticsearch现有集群扩展节点 – 无敌牛 欢迎参观我的个人博客:无敌牛 – 技术/著作/典籍/分享等 给现有的 elasticsearch 集群扩展节点比较容易,已有的集群不需要做任何修改,也不用对服务做任何处理,只需…...
XML Group端口详解
在XML数据映射过程中,经常需要对数据进行分组聚合操作。例如,当处理包含多个物料明细的XML文件时,可能需要将相同物料号的明细归为一组,或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码,增加了开…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
深度学习在微纳光子学中的应用
深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向: 逆向设计 通过神经网络快速预测微纳结构的光学响应,替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…...
[2025CVPR]DeepVideo-R1:基于难度感知回归GRPO的视频强化微调框架详解
突破视频大语言模型推理瓶颈,在多个视频基准上实现SOTA性能 一、核心问题与创新亮点 1.1 GRPO在视频任务中的两大挑战 安全措施依赖问题 GRPO使用min和clip函数限制策略更新幅度,导致: 梯度抑制:当新旧策略差异过大时梯度消失收敛困难:策略无法充分优化# 传统GRPO的梯…...
conda相比python好处
Conda 作为 Python 的环境和包管理工具,相比原生 Python 生态(如 pip 虚拟环境)有许多独特优势,尤其在多项目管理、依赖处理和跨平台兼容性等方面表现更优。以下是 Conda 的核心好处: 一、一站式环境管理:…...
FFmpeg 低延迟同屏方案
引言 在实时互动需求激增的当下,无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作,还是游戏直播的画面实时传输,低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架,凭借其灵活的编解码、数据…...
Python爬虫实战:研究feedparser库相关技术
1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的信息资源。RSS(Really Simple Syndication)作为一种标准化的信息聚合技术,被广泛用于网站内容的发布和订阅。通过 RSS,用户可以方便地获取网站更新的内容,而无需频繁访问各个网站。 然而,互联网…...
五年级数学知识边界总结思考-下册
目录 一、背景二、过程1.观察物体小学五年级下册“观察物体”知识点详解:由来、作用与意义**一、知识点核心内容****二、知识点的由来:从生活实践到数学抽象****三、知识的作用:解决实际问题的工具****四、学习的意义:培养核心素养…...
Redis的发布订阅模式与专业的 MQ(如 Kafka, RabbitMQ)相比,优缺点是什么?适用于哪些场景?
Redis 的发布订阅(Pub/Sub)模式与专业的 MQ(Message Queue)如 Kafka、RabbitMQ 进行比较,核心的权衡点在于:简单与速度 vs. 可靠与功能。 下面我们详细展开对比。 Redis Pub/Sub 的核心特点 它是一个发后…...
【分享】推荐一些办公小工具
1、PDF 在线转换 https://smallpdf.com/cn/pdf-tools 推荐理由:大部分的转换软件需要收费,要么功能不齐全,而开会员又用不了几次浪费钱,借用别人的又不安全。 这个网站它不需要登录或下载安装。而且提供的免费功能就能满足日常…...
