里氏代换原则详解

里氏代换原则详解里氏代换原则定义 里氏代换原则 LiskovSubsti 面向对象设计的基本原则之一 里氏代换原则中说 任何基类可以出现的地方 子类一定可以出现 LSP 是继承复用的基石 只有当衍生类可以替换掉基类 软件单位的功能不受到影响时 基类才能真正被复用 而衍生类也能够在基类的基础上增加新的行为 里氏代换原则是对 开 闭 原则的补充 实现 开 闭 原则的关键步骤就是抽象化 而基类与子类的继承关系就是抽象化的具体实现 所以里氏代换原则是对实现抽象化的具体步骤的规

里氏代换原则

定义

​里氏代换原则(Liskov Substitution Principle LSP)面向对象设计的基本原则之一。

里氏代换原则中说,任何基类可以出现的地方,子类一定可以出现。

LSP是继承复用的基石,只有当衍生类可以替换掉基类,软件单位的功能不受到影响时,基类才能真正被复用,而衍生类也能够在基类的基础上增加新的行为。

​ 里氏代换原则是对“开-闭”原则的补充。实现“开-闭”原则的关键步骤就是抽象化。而基类与子类的继承关系就是抽象化的具体实现,所以里氏代换原则是对实现抽象化的具体步骤的规范。

原则

第一点

子类必须实现父类的抽象方法,但不得重写(覆盖)父类的非抽象(已实现)方法。

class Foo { 
    public void cal(int num1, int num2) { 
    int value = num1 + num2; System.out.println("父类计算结果: " + value); } } class Son extends Foo { 
    public void cal(int num1, int num2) { 
    int value = num1 - num2; System.out.println("子类计算结果:" + value); } } class Cal{ 
    public static void main(String[] args) { 
    Foo foo = new Foo(); foo.cal(2,1); Son son = new Son(); son.cal(2,1); } } 

​ 在类的继承中,我们的父类定义好的方法,并不会强制要求其子类必须完全遵守该方法的实现规则。子类是可以修改它继承自父类的任意方法的。 在本例中,父类的本意是想要定义一个两数相加的方法,但是子类继承该方法后却修改为减法,并且也成功了。子类这样操作后,会对整个继承体系造成破坏。当你想把使用父类的地方替换为其子类时,会发现原来的正常的功能现在出现问题了。

第二点

当子类需要重载父类中的方法的时候,子类方法的形参(入参)要比父类方法输入的参数更宽松(范围更广)。

class Foo { 
    public void method(List arrayList) { 
    System.out.println("父类方法执行"); } } class Son extends Foo { 
    public void method(ArrayList list) { 
    System.out.println("子类方法执行" ); } } class Cal{ 
    public static void main(String[] args) { 
    ArrayList list = new ArrayList(); Foo foo = new Foo(); Son son = new Son(); System.out.println("使用父类对象调用的结果:"); foo.method(list); System.out.println("将父类对象替换为子类对象调用结果"); son.method(list); } } //输出 使用父类对象调用的结果: 父类方法执行 将父类对象替换为子类对象调用结果 子类方法执行 

​ 我们的本意是希望对象替换后还执行原来的方法的,可结果却发生变化了。

修改

class Foo { 
    public void method(ArrayList arrayList) { 
    System.out.println("父类方法执行"); } } class Son extends Foo { 
    //重载了父类的method,并且方法入参比父类的入仓范围更广 public void method(List list) { 
    System.out.println("子类方法执行" ); } } 

第三点

重写或者实现父类方法的时候,方法的返回值可以被缩小,但是不能放大。 正例:

 class Foo { 
    public List getList() { 
    return new ArrayList(); } } class Son extends Foo { 
    public ArrayList getList() { 
    return new ArrayList(); } } 

反例:

class Foo { 
    public ArrayList getList() { 
    return new ArrayList(); } } class Son extends Foo { 
    public List getList() { 
    return new ArrayList(); } } 

​ 如果我们试图在子类中放大,重写或实现来自父类方法的返回值时,代码会报错,连基本的编译器都无法通过。

第四点

子类可以拥有自己独特的方法或属性

class Foo { 
    public void cal(int num1, int num2) { 
    int value = num1 + num2; System.out.println("父类计算结果: " + value); } } class Son extends Foo { 
    public void cal(int num1, int num2) { 
    int value = num1 - num2; System.out.println("子类计算结果:" + value); } public void cal2(int num1, int num2) { 
    int value = num1 + num2 +num2; System.out.println("子类计算结果:" + value); } } 

总结

​ 通过上面的描述相信大家都对里氏替换原则有了一个基本的概念,其实它就是告诉我们在继承中需要注意什么问题和遵守什么规则。

​ 然而在实际开发中我们在很多时候还是会违背该原则的,虽然表面上没有什么特别大的问题,但是这样做会大大增加代码的出错率。我们编写代码时不光要考虑怎么实现该功能,程序的健壮性和后期的扩展以及移植都是需要考虑到的。只有这样做才可以使我们的程序更加优秀。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/220924.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月17日 下午7:20
下一篇 2026年3月17日 下午7:20


相关推荐

  • 外链图床-PicGo-Gitee

    外链图床-PicGo-Gitee前言刚开始写建立博客的时候,头疼过图片该如何处理:直接放github的仓库里吧。可是随着图片越来越多,每次pull、clone都需要耗费大量的时间。而且github的仓库加载很慢,影响博客的流畅性。外链图床吧。可是国内基本上没有什么好用的免费图床,七牛云之类的都需要收费的。国外的图床加载速度又是特别的慢。自己搭建。图床搭建所使用的工具为PicGo、Gitee、Typora.在这之前,我一般在VSCode上面写Markdown文档,但是现在开始添加图片到博客中去,转而使用Typora,以便

    2022年6月21日
    39
  • 盘点 6 个开源的音乐播放器!

    盘点 6 个开源的音乐播放器!盘点几个开源的音乐播放器,在这些项目中你可以学习到React、Vue、Kotlin等相关的技术栈。有的是仿当前比较火的音乐应用,而有的是开发者自主开发。本期推荐开源项目目录:1.仿QQ音乐2.网易云音乐3.Material主题音乐播放器4.不太现代的音乐播放器5.专注于免费流媒体资源的桌面音乐播放器01仿QQ音乐模仿QQ音乐网页版界面,采用fl…

    2022年6月26日
    74
  • 华为c++面试题和答案汇总_C语言经典笔试题

    华为c++面试题和答案汇总_C语言经典笔试题《几道华为经典C语言面试题》由会员分享,可在线阅读,更多相关《几道华为经典C语言面试题(6页珍藏版)》请在人人文库网上搜索。1、1、找错voidtest1()charstring10;char*str1=;strcpy(string,str1);这里string数组越界,因为字符串长度为10,还有一个结束符0。所以总共有11个字符长度。string数组大小为10,这里越界了。PS:使用st…

    2022年8月28日
    6
  • java实现国密SM4加密「建议收藏」

    java实现国密SM4加密「建议收藏」前言最近世界政治影响,我国也开始要求算法的使用,以避免来自外国的黑客入侵。我们在使用加密算法时,有必要选择使用国密算法进行加密一、国密SM4是什么? 国密即国家密码局认定的国产密码算法。 主要有SM1,SM2,SM3,SM4。密钥长度和分组长度均为128位。 SM1为对称加密。其加密强度与AES相当。该算法不公开,调用该算法时,需要通过加密芯片的接口进行调用。 SM2为非对称加密,基于ECC。该算法已公开。由于该算法基于ECC,故其签名速度与秘钥生成速度都快于RSA ECC2.

    2022年10月5日
    6
  • Polar码概述[通俗易懂]

    Polar码概述[通俗易懂]2016年11月18日,在美国内华达州里诺召开的3GPPRAN1#87次会议,确定PolarCode作为5GeMBB(增强移动宽带)场景下控制信道编码方案。2008年,ErdalArikan在国际信息论ISIT会议上首次提出了信道极化(ChannelPolarization)的概念;2009年在“IEEETransactiononInformationTheory”期刊上

    2025年7月29日
    5
  • batchnorm原理及代码详解(笔记2)

    batchnorm原理及代码详解(笔记2)Batchnorm原理详解前言:Batchnorm是深度网络中经常用到的加速神经网络训练,加速收敛速度及稳定性的算法,可以说是目前深度网络必不可少的一部分。本文旨在用通俗易懂的语言,对深度学习的常用算法–batchnorm的原理及其代码实现做一个详细的解读。本文主要包括以下几个部分。Batchnorm主要解决的问题 Batchnorm原理解读 Batchnorm的优点 Batc…

    2022年5月28日
    38

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号