【设计原则】里氏替换原则(LSP):构建稳健继承体系的黄金法则

一、什么是里氏替换原则?

里氏替换原则(Liskov Substitution Principle, LSP)是面向对象设计SOLID原则中的"L",由Barbara Liskov在1987年提出。其核心定义为:

所有引用基类(父类)的地方必须能透明地使用其子类的对象

这意味着:

  • 子类必须完全实现父类的抽象方法
  • 子类可以扩展父类功能但不能改变原有行为
  • 子类方法的前置条件不应强于父类
  • 子类方法的后置条件不应弱于父类

二、为什么需要LSP?

  1. 保证继承关系的正确性
  2. 提高代码的可维护性
  3. 增强系统的可扩展性
  4. 降低单元测试的复杂度

三、经典违反案例:矩形与正方形问题

// 基类:矩形
public class Rectangle
{// 矩形的宽度属性public virtual int Width { get; set; }// 矩形的高度属性public virtual int Height { get; set; }// 计算矩形的面积public int Area => Width * Height;
}// 子类:正方形
public class Square : Rectangle
{// 重写Width属性,确保宽度和高度始终相等public override int Width{set { base.Width = base.Height = value; }}// 重写Height属性,确保高度和宽度始终相等public override int Height {set { base.Width = base.Height = value; }}
}// 使用场景:面积计算器
public class AreaCalculator
{// 计算矩形面积的方法public void Calculate(Rectangle rect){// 设置宽度为5rect.Width = 5;// 设置高度为4rect.Height = 4;// 输出期望面积和实际面积Console.WriteLine($"期望面积20,实际得到:{rect.Area}");}
}// 调用时会出现问题
new AreaCalculator().Calculate(new Square());  // 输出16而不是20

问题分析
Square改变了Rectangle的基本行为约定,导致父类替换时出现意外结果,违反了LSP。

四、正确的设计实践

方案1:通过接口分离

// 定义形状接口
public interface IShape
{// 面积属性int Area { get; }
}// 矩形类实现IShape接口
public class Rectangle : IShape
{// 宽度属性public int Width { get; set; }// 高度属性public int Height { get; set; }// 计算面积public int Area => Width * Height;
}// 正方形类实现IShape接口
public class Square : IShape
{// 边长属性public int SideLength { get; set; }// 计算面积public int Area => SideLength * SideLength;
}

方案2:使用抽象类

// 定义抽象形状类
public abstract class Shape
{// 抽象面积属性public abstract int Area { get; }
}// 矩形类继承Shape
public class Rectangle : Shape
{// 宽度属性public int Width { get; set; }// 高度属性public int Height { get; set; }// 实现面积计算public override int Area => Width * Height;
}// 正方形类继承Shape
public class Square : Shape
{// 边长属性public int SideLength { get; set; }// 实现面积计算public override int Area => SideLength * SideLength;
}

五、LSP的关键检查点

  1. 方法签名一致性

    // 父类:鸟
    public class Bird {// 飞的方法public virtual void Fly() { /*...*/ }
    }// 违反LSP的子类:企鹅
    public class Penguin : Bird {// 重写Fly方法,抛出异常public override void Fly() {throw new NotSupportedException();}
    }
    

    解决方案:建立IFlyable接口

  2. 前置条件不强于父类

    // 父类
    public virtual void SetTemperature(int temp) {// 接受0-100
    }// 违反LSP的子类
    public override void SetTemperature(int temp) {if(temp < 10) throw new ArgumentException(); // 加强限制//...
    }
    
  3. 后置条件不弱于父类

    // 父类方法保证返回正数
    public virtual int Calculate() {return Math.Abs(result);
    }// 违反LSP的子类
    public override int Calculate() {return result; // 可能返回负数
    }
    

六、C#中的实现建议

  1. 使用"override"关键字确保正确重写
  2. 密封基类方法防止意外修改
    public class Vehicle {// 密封Start方法,防止子类修改public sealed override void Start() { /* 基础实现 */ }
    }
    
  3. 接口默认实现(C#8.0+)
    public interface IWorker {// 默认实现Work方法void Work() => Console.WriteLine("Working...");
    }
    

七、单元测试验证LSP

使用NUnit进行契约测试:

[TestFixture]
public class LspTests {[Test]public void TestRectangleSubstitution() {// 创建形状列表var shapes = new List<Shape> { new Rectangle(), new Square() };// 遍历每个形状foreach(var shape in shapes) {// 设置宽度和高度shape.Width = 5;shape.Height = 4;// 断言面积是否为20Assert.That(shape.Area, Is.EqualTo(20));}}
}

八、最佳实践总结

  1. 优先使用组合而非继承
  2. 保持继承层次扁平化
  3. 使用设计模式:
    • 策略模式
    • 模板方法模式
    • 装饰器模式
  4. 定期进行代码审查
  5. 编写契约测试

九、现实应用场景

  1. 支付系统:
    // 抽象支付提供者
    public abstract class PaymentProvider {// 抽象支付方法public abstract void ProcessPayment(decimal amount);
    }// 信用卡支付实现
    public class CreditCardPayment : PaymentProvider { /*...*/ }// PayPal支付实现
    public class PayPalPayment : PaymentProvider { /*...*/ }
    
  2. 日志系统:
    // 日志接口
    public interface ILogger {// 日志记录方法void Log(string message);
    }// 文件日志实现
    public class FileLogger : ILogger { /*...*/ }// 数据库日志实现
    public class DatabaseLogger : ILogger { /*...*/ }
    

遵循LSP能够创建出更健壮、更易维护的系统架构。记住:好的继承关系应该表现为"is-a"的关系,而不是"is-like-a"。当发现子类需要修改父类核心行为时,这往往是一个设计需要改进的信号。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/71214.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于Electron的应用程序安全测试基础 — 提取和分析.asar文件的案例研究

目录&#xff1a; 4.4. 案例研究 4.4.2. 情况描述 4.4.3. 信息收集 4.4.3.2. 检查隐藏目录&#xff08;点目录&#xff09;的可能性 4.4.3.3. 使用 DB Browser for SQLite 打开 .db 文件 4.4.3.4. 寻找加密算法 4.4.3.5. 找到加密算法 4.4.3.6. 理解加密流程 4.4.3.7. 找到“Ke…

【Delphi】如何解决使用webView2时主界面置顶,而导致网页选择文件对话框被覆盖问题

一、问题描述&#xff1a; 在Delphi 中使用WebView2控件&#xff0c;如果预先把主界面置顶&#xff08;Self.FormStyle : fsStayOnTop;&#xff09;&#xff0c;此时&#xff0c;如果在Web页面中有使用&#xff08;<input type"file" id"fileInput" acc…

ASP.NET Core 3.1 修改个别API返回JSON序列化格式

ASP.NET Core 3.0及之后的版本中&#xff0c;默认的JSON格式化器是基于System.Text.Json的。返回json格式采用camelCase&#xff08;第一个单词首字母小写&#xff0c;后面单词首字母大写&#xff09;。如果想改为PascalCase&#xff0c;可以全局设置PropertyNamingPolicy nul…

有关Java中的集合(2):Map<T>(底层源码分析)

学习目标 核心掌握Map集合 1.Map<K,V> ● 实现了Map接口的集合对象的集合元素&#xff1a; 成对的值 key-value 键值对 ● key对象是不能重复的. value可以重复。 ● 核心: 根据key获得value。 1.1 层级 public interface Map<K, V> {}1.2 常用方法 1.3 使用方法…

windows电脑上安装llama-factory实现大模型微调

一、安装环境准备 这是官方给的llama-factory安装教程&#xff0c;安装 - LLaMA Factory&#xff0c;上面介绍了linux系统上以及windows系统上如何正确安装。大家依照安装步骤基本能够完成安装&#xff0c;但是可能由于缺少经验或者相关的知识导致启动webUi界面运行相应内容时…

每日一题——接雨水

接雨水问题详解 问题描述 给定一个非负整数数组 height&#xff0c;表示每个宽度为 1 的柱子的高度图。计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 示例 示例 1&#xff1a; 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;6 解释&#…

Ubuntu 创建新用户及设置权限

1、新建用户 sudo adduser username 其中username是你要创建的用户的用户名&#xff0c;然后设置密码和相关信息就可以了 2、给新用户sudo权限 新创建的用户没有root权限&#xff0c;我们执行以下命令给用户sudo权限 sudo usermod -a -G adm username sudo usermod -a -G s…

商米科技前端工程师(base上海)内推

1.根据原型或高保真设计&#xff0c;开发web、H5、小程序等类型的前端应用&#xff1b; 2.在指导下&#xff0c;高质量完成功能模块的开发&#xff0c;并负责各功能模块接口设计工作&#xff1b; 3.负责产品及相关支撑系统的开发及维护工作&#xff0c;不断的优化升级&#x…

HTTP四次挥手是什么?

四次挥手&#xff0c;这是TCP协议用来关闭连接的过程。四次挥手是确保两个主机之间能够安全、可靠地关闭连接的重要机制。我会用简单易懂的方式来讲解&#xff0c;帮助你理解它的原理和过程。 1. 什么是四次挥手&#xff1f; 定义 四次挥手是TCP协议用来关闭连接的过程。它通…

使用 REINFORCE 算法强化梯度策略

一、整体概述 此代码利用 REINFORCE 算法&#xff08;一种基于策略梯度的强化学习算法&#xff09;来解决 OpenAI Gym 中的 CartPole-v1 环境问题。CartPole-v1 环境的任务是控制一个小车&#xff0c;使连接在小车上的杆子保持平衡。代码通过构建一个神经网络作为策略网络&…

使用Python自动生成图文并茂的网页分析报告

在数据分析中&#xff0c;不管是市场研究还是科学分析&#xff0c;经常需要使用Python进行数据分析并生成图表报告。一般使用Python生成和展示图表时都是使用matplotlib 库生成静态图片文件&#xff0c;这种方式不便之处是不方便跟动态文字段落结合在一起&#xff0c;也不方便分…

【iOS】小蓝书学习(七)

小蓝书学习&#xff08;七&#xff09; 前言第47条&#xff1a;熟悉系统框架第48条&#xff1a;多用枚举块&#xff0c;少用for循环第50条&#xff1a;构建缓存使选用NSCache而非NSDictionary第51条&#xff1a;精简initialize与load的实现代码第52条&#xff1a;别忘了NSTimer…

C语言复习4:有关数组的基础常见算法

# 数组的常见算法 - 查找算法 1. 基本查找/顺序查找 2. 二分查找/折半查找 3. 插值查找 4. 分块查找 5. 哈希查找 6. 树表查找 7. 斐波那契查找 - 排序算法&#xff08;顾名思义&#xff0c;就是把没有顺序的…

Ollama 的庐山真面目

Ollama 运行方式分析 本地推理条件&#xff08;GPU/CPU/RAM&#xff09;&#xff1a;Ollama 支持在本地电脑进行大模型推理&#xff0c;但需要满足一定的硬件条件。一般来说&#xff0c;GPU 有助于加速推理&#xff0c;特别是显存较大的 GPU 能够加载更大的模型&#xff1b;如果…

SyntaxError: positional argument follows keyword argument

命令行里面日常练手爬虫不注意遇到的问题&#xff0c;报错说参数位置不正确 修改代码后&#xff0c;运行如下图&#xff1a; 结果&#xff1a; 希望各位也能顺利解决问题&#xff0c;祝你好运&#xff01;

drawDB:一款免费数据库设计工具

drawDB 是一款基于 Web 的免费数据库设计工具&#xff0c;通过拖拽、复制、粘贴等方式进行数据库建模设计&#xff0c;同时可以生成相应的 SQL 脚本。 功能特性 drawDB 目前可以支持 MySQL、MariaDB、PostgreSQL、SQL Server 以及 SQLite 数据库&#xff0c;核心功能包括&…

使用SPI总线与外部传感器通信,使用ECU抽象

MCAL SPI驱动示例 首先,MCAL层提供了针对特定微控制器的SPI驱动实现。以下是一个简化的MCAL SPI驱动API的例子: // MCAL SPI driver interface void Spi_Init(const Spi_ConfigType* Config); Std_ReturnType Spi_Transmit(uint8 *DataBufferPtr, uint8 Length); Std_Retur…

FPGA开发,使用Deepseek V3还是R1(9):FPGA的全流程(详细版)

以下都是Deepseek生成的答案 FPGA开发&#xff0c;使用Deepseek V3还是R1&#xff08;1&#xff09;&#xff1a;应用场景 FPGA开发&#xff0c;使用Deepseek V3还是R1&#xff08;2&#xff09;&#xff1a;V3和R1的区别 FPGA开发&#xff0c;使用Deepseek V3还是R1&#x…

Conda 环境搭建实战:从基础到进阶

在当今复杂多变的软件开发与数据科学领域&#xff0c;拥有一个稳定、可复现且易于管理的开发环境是项目成功的基石。Conda 作为一款强大的跨平台环境管理与包管理工具&#xff0c;为开发者提供了便捷高效的环境搭建与依赖管理解决方案。本文将深入探讨 Conda 环境搭建的实战技巧…

Hive-05之查询 分组、排序、case when、 什么情况下Hive可以避免进行MapReduce

一、目标 掌握hive中select查询语句中的基本语法掌握hive中select查询语句的分组掌握hive中select查询语句中的join掌握hive中select查询语句中的排序 二、要点 1. 基本查询 注意 SQL 语言大小写不敏感SQL 可以写在一行或者多行关键字不能被缩写也不能分行各子句一般要分行…