里氏替换原则

一 里氏替换原则(Liskov Substitution Principle, LSP)

要么别继承,如果继承,就要支持父类所有方法


一句话核心思想

"子类可以替换父类,并且程序不会出错。"

(就像你爸的手机你能用,你的手机你爸也能用,谁用都不会出问题!)


通俗解释(现实例子)

假设你有一个 "鸟" 类,它有个方法 "飞"

java 复制代码
class Bird {
    void fly() {
        System.out.println("鸟儿飞翔~");
    }
}

然后你写了个 "企鹅" 类继承 "鸟"

java 复制代码
class Penguin extends Bird {
    @Override
    void fly() {
        throw new RuntimeException("企鹅不会飞!"); // 违背LSP!
    }
}

问题来了

  • 如果别人调用 Bird bird = new Penguin(); bird.fly();程序崩溃!
  • 这就是违反LSP,因为 企鹅(子类)不能安全替换鸟(父类)

正确做法(遵守LSP)

  1. 要么别继承 (企鹅本就不该继承"会飞的鸟"):

    java 复制代码
    class Bird { } // 普通鸟
    class FlyingBird extends Bird { void fly() { ... } } // 会飞的鸟
    class Penguin extends Bird { } // 企鹅不会飞,但不继承FlyingBird
  2. 要么子类别破坏父类行为如果继承,就要支持父类所有方法 ):

    java 复制代码
    class Penguin extends Bird {
        @Override
        void fly() {
            System.out.println("企鹅不会飞,但可以游泳!"); // 不报错,而是优雅处理
        }
    }

关键记忆点(LSP核心规则)

  1. 子类必须完全支持父类的功能(不能"阉割"父类方法)。
  2. 子类可以扩展功能,但不能改变父类的初衷(比如父类是"飞",子类不能改成"跑")。
  3. 调用父类的地方,换成子类也不会报错(这是黄金检验标准!)。

实际代码中如何遵守?

正确示范

  • 父类 Fileread() 方法,子类 PDFFileTXTFile 都实现 read(),但内部逻辑不同。
  • 父类 Listadd() 方法,子类 UnmodifiableList 直接禁止修改(抛出明确异常,而非隐藏错误)。

错误示范

  • 父类 RectanglesetWidth()setHeight(),子类 Square 改成只用一个边长(违反数学逻辑)。
  • 父类 Databaseconnect(),子类 MockDatabaseconnect() 直接返回 null(应返回模拟连接)。

一句话总结

"子类要像充话费送的亲儿子,不能像捡来的干儿子------该有的功能一个不能少,但可以更牛逼!"

二 为什么继承xxxFragment违反里氏替换原则(LSP):

  1. LSP核心原则:
  • 子类应该能够替换父类,而不改变程序的正确性
  • 子类可以扩展父类的功能,但不应该改变父类的功能
  1. xxxFragment包含特定的空调业务逻辑:
kotlin 复制代码
class xxxFragment : BasexxxFragment() {
    private val frontFragment by lazy { FxxFragment() }
    private val rearFragment by lazy { RxxFragment() }
    
    private fun setupFragments() {
        // 空调特有的前后排切换逻辑
        childFragmentManager.beginTransaction()
            .add(R.id.fragmentContainer, rearFragment)
            .hide(rearFragment)
            .add(R.id.fragmentContainer, frontFragment)
            .commit()
    }
}
  1. 违反LSP的情况:
  • 如果其他Fragment(如MassageFragment)继承AirConditionFragment
  • 它会继承到不需要的空调UI逻辑
  • 必须重写或忽略父类的空调相关方法
  • 这破坏了父类的行为预期

正确的做法是创建SharedViewModelFragment作为基类,因为:

  1. 只包含共享的ViewModel逻辑

  2. 不包含具体业务实现

  3. 子类可以安全扩展而不违反LSP

三 里氏替换原则的具体含义可以概括为以下几点

1‌.子类必须能够替换其基类‌:在任何基类可以出现的地方,子类都可以出现,并且不会改变程序的行为。

‌2.子类扩展而非修改‌:子类可以增加新的行为,但不能改变或删除父类已有的功能。这意味着子类可以添加新的方法或重载父类的方法,但不能重写父类的非抽象方法。

3‌. 对开闭原则的补充‌:里氏替换原则是对开闭原则(Open-Closed Principle)的补充,强调通过继承和抽象化来实现软件的可扩展性和可维护性。

参考:

https://blog.csdn.net/lilinhai548/article/details/141287849

相关推荐
鹏多多.1 小时前
flutter-使用device_info_plus获取手机设备信息完整指南
android·前端·flutter·ios·数据分析·前端框架
来来走走6 小时前
Flutter开发 网络请求
android·flutter
独行soc13 小时前
2025年渗透测试面试题总结-18(题目+回答)
android·python·科技·面试·职场和发展·渗透测试
雨白13 小时前
登录和授权:Cookie与Authorization Header机制详解
android
Frank_HarmonyOS14 小时前
【Android -- 多线程】Handler 消息机制
android
一条上岸小咸鱼15 小时前
Kotlin 基本数据类型(一):概述及分类
android·kotlin
没盐水菠萝16 小时前
Android - 动态切换桌面图标
android
AI 嗯啦16 小时前
SQL详细语法教程(三)mysql的函数知识
android·开发语言·数据库·python·sql·mysql
跨界混迹车辆网的Android工程师17 小时前
adb 发送广播
android
超勇的阿杰19 小时前
gulimall项目笔记:P54三级分类拖拽功能实现
android·笔记