【设计模式】SOLID 设计原则概述

news2025/3/25 22:02:28

SOLID 是面向对象设计中的五大原则,不管什么面向对象的语言, 这个准则都很重要,如果你没听说过,赶紧先学一下。它可以提高代码的可维护性可扩展性可读性,使代码更加健壮、易于测试和扩展。SOLID 代表以下五个设计原则:

  1. S - 单一职责原则(Single Responsibility Principle, SRP)
  2. O - 开闭原则(Open/Closed Principle, OCP)
  3. L - 里氏替换原则(Liskov Substitution Principle, LSP)
  4. I - 接口隔离原则(Interface Segregation Principle, ISP)
  5. D - 依赖倒置原则(Dependency Inversion Principle, DIP)

1. 单一职责原则(SRP)

一个类应该仅有一个引起它变化的原因
一个类应该仅负责一个功能,否则后期修改代码时,可能会影响其他无关功能。

❌ 违反 SRP 的例子

class Report {
public:
    void generateReport() { /* 生成报表 */ }
    void printReport() { /* 打印报表 */ }
    void saveToFile() { /* 保存到文件 */ }
};

问题:
Report 负责 生成报表打印报表保存报表,违反了 SRP。

✅ 遵循 SRP 的改进

class Report {
public:
    void generateReport() { /* 生成报表 */ }
};

class ReportPrinter {
public:
    void printReport(Report& report) { /* 打印报表 */ }
};

class ReportSaver {
public:
    void saveToFile(Report& report) { /* 保存到文件 */ }
};

改进点:

  • Report 只负责生成报表
  • ReportPrinter 负责打印
  • ReportSaver 负责存储

这样每个类的变更都不会影响其他功能,符合 SRP。


2. 开闭原则(OCP)

软件实体(类、模块、函数)应该对扩展开放,对修改关闭。
即:新增功能时,应该通过扩展代码,而不是修改已有代码

❌ 违反 OCP 的例子

class PaymentProcessor {
public:
    void processPayment(std::string paymentType) {
        if (paymentType == "CreditCard") {
            // 处理信用卡支付
        } else if (paymentType == "PayPal") {
            // 处理 PayPal 支付
        }
    }
};

问题:

  • 如果新增 Apple Pay,需要修改 processPayment(),违反 OCP。
  • 代码越复杂,修改的风险越高。

✅ 遵循 OCP 的改进

class Payment {
public:
    virtual void pay() = 0;
    virtual ~Payment() = default;
};

class CreditCardPayment : public Payment {
public:
    void pay() override { /* 处理信用卡支付 */ }
};

class PayPalPayment : public Payment {
public:
    void pay() override { /* 处理 PayPal 支付 */ }
};

class PaymentProcessor {
public:
    void processPayment(Payment& payment) {
        payment.pay();
    }
};

改进点:

  • 新增支付方式时,不需要修改 PaymentProcessor,只需新增一个类(符合 OCP)。
  • 通过 多态 使代码更加灵活。

3. 里氏替换原则(LSP)

子类必须能够替换基类,并且不会破坏程序的正确性

❌ 违反 LSP 的例子

class Bird {
public:
    virtual void fly() { /* 飞行逻辑 */ }
};

class Penguin : public Bird {
public:
    void fly() override {
        throw std::runtime_error("企鹅不会飞!");
    }
};

问题:

  • Penguin 继承了 Bird,但企鹅不会飞!
  • Penguin::fly() 违背了父类的逻辑,可能导致程序崩溃。

✅ 遵循 LSP 的改进

class Bird {
public:
    virtual void move() = 0;
};

class FlyingBird : public Bird {
public:
    void move() override { /* 飞行逻辑 */ }
};

class Penguin : public Bird {
public:
    void move() override { /* 企鹅走路 */ }
};

改进点:

  • 抽象出 FlyingBirdPenguin,使 Penguin 不继承 fly(),从而避免违反 LSP。

4. 接口隔离原则(ISP)

不应该强迫类实现它们不需要的接口
即:一个接口不应该承担过多职责,而应该拆分成多个专门的接口

❌ 违反 ISP 的例子

class Worker {
public:
    virtual void work() = 0;
    virtual void eat() = 0;
};
class Robot : public Worker {
public:
    void work() override { /* 机器人工作 */ }
    void eat() override { throw std::runtime_error("机器人不吃饭!"); }
};

问题:

  • Robot 不需要 eat(),但仍然要实现它,违反 ISP

✅ 遵循 ISP 的改进

class Workable {
public:
    virtual void work() = 0;
};

class Eatable {
public:
    virtual void eat() = 0;
};

class Human : public Workable, public Eatable {
public:
    void work() override { /* 人工作 */ }
    void eat() override { /* 人吃饭 */ }
};

class Robot : public Workable {
public:
    void work() override { /* 机器人工作 */ }
};

改进点:

  • Worker 拆分成 WorkableEatable,避免不必要的实现。

5. 依赖倒置原则(DIP)

高层模块不应该依赖低层模块,而应该依赖于抽象(接口)

❌ 违反 DIP 的例子

class LEDLight {
public:
    void turnOn() { /* 打开 LED 灯 */ }
};

class Switch {
private:
    LEDLight light;
public:
    void operate() { light.turnOn(); }
};

问题:

  • Switch 直接依赖 LEDLight,如果要支持 白炽灯,必须修改 Switch,违反 OCP 和 DIP

✅ 遵循 DIP 的改进

class Light {
public:
    virtual void turnOn() = 0;
    virtual ~Light() = default;
};

class LEDLight : public Light {
public:
    void turnOn() override { /* 打开 LED 灯 */ }
};

class IncandescentLight : public Light {
public:
    void turnOn() override { /* 打开白炽灯 */ }
};

class Switch {
private:
    Light& light;
public:
    Switch(Light& l) : light(l) {}
    void operate() { light.turnOn(); }
};

改进点:

  • Switch 依赖 Light 接口,不依赖具体的 LEDLight,符合 DIP
  • 以后要支持新灯泡 不修改 Switch 代码。

总结

原则含义好处
SRP一个类只做一件事降低耦合,提高可维护性
OCP类应对扩展开放,对修改关闭增加功能时不修改已有代码
LSP子类可以替换父类,不影响功能确保继承不会破坏系统
ISP接口应该小而精,不要强迫实现不需要的功能降低实现负担,提高灵活性
DIP依赖接口,不依赖具体实现提高可扩展性,降低耦合

结论

遵循 SOLID 原则可以写出更好的代码,使系统更易扩展、维护和测试。在设计类和模块时,应尽量减少耦合,提高可复用性,避免不必要的修改。

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

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

相关文章

others-rustdesk远程

title: others-rustdesk远程 categories: Others tags: [others, 远程] date: 2025-03-19 10:19:34 comments: false mathjax: true toc: true others-rustdesk远程, 替代 todesk 的解决方案 前篇 官方 服务器 - https://rustdesk.com/docs/zh-cn/self-host/rustdesk-server-o…

C++基础 [八] - list的使用与模拟实现

目录 list的介绍 List的迭代器失效问题 List中sort的效率测试 list 容器的模拟实现思想 模块分析 作用分析 list_node类设计 list 的迭代器类设计 迭代器类--存在的意义 迭代器类--模拟实现 模板参数 和 成员变量 构造函数 * 运算符的重载 运算符的重载 -- 运…

使用excel.EasyExcel实现导出有自定义样式模板的excel数据文件,粘贴即用!!!

客户要求导出的excel文件是有好看格式的,当然本文举例模板文件比较简单,内容丰富的模板可以自行设置,话不多说,第一步设置一个"好看"的excel文件模板 上面要注意的地方是{.变量名} ,这里的变量名对应的就是…

Spring Boot 集成 Elasticsearch怎样在不启动es的情况下正常启动服务

解释 在spingboot 集成es客户端后,每当服务启动时,服务默认都会查看es中是否已经创建了对应的索引,如果没有索引则创建。基于上面的规则我们可以通过配置不自动创建索引来达到在没有es服务的情况下正常启动服务。 解决办法 在entity类的Docu…

JVM常见概念之条件移动

问题 当我们有分支频率数据时,有什么有趣的技巧可以做吗?什么是条件移动? 基础知识 如果您需要在来自一个分支的两个结果之间进行选择,那么您可以在 ISA 级别做两件不同的事情。 首先,你可以创建一个分支&#xff…

Android AI ChatBot-v1.6.3-28-开心版[免登录使用GPT-4o和DeepSeek]

Android AI ChatBot- 链接:https://pan.xunlei.com/s/VOLi1Ua071S6QZBGixcVL5eeA1?pwdp3tt# 免登录使用GPT-4o和DeepSeek

集成学习(上):Bagging集成方法

一、什么是集成学习? 在机器学习的世界里,没有哪个模型是完美无缺的。就像古希腊神话中的"盲人摸象",单个模型往往只能捕捉到数据特征的某个侧面。但当我们把多个模型的智慧集合起来,就能像拼图一样还原出完整的真相&a…

DeepSeek R1 本地部署指南 (3) - 更换本地部署模型 Windows/macOS 通用

0.准备 完成 Windows 或 macOS 安装: DeepSeek R1 本地部署指南 (1) - Windows 本地部署-CSDN博客 DeepSeek R1 本地部署指南 (2) - macOS 本地部署-CSDN博客 以下内容 Windows 和 macOS 命令执行相同: Windows 管理员启动:命令提示符 CMD ma…

【TI MSPM0】Timer学习

一、计数器 加法计数器:每进入一个脉冲,就加一减法计算器:每进入一个脉冲,就减一 当计数器减到0,触发中断 1.最短计时时间 当时钟周期为1khz时,最短计时时间为1ms,最长计时时间为65535ms 当时…

Windows部署deepseek R1训练数据后通过AnythingLLM当服务器创建问答页面

如果要了解Windows部署Ollama 、deepseek R1请看我上一篇内容。 这是接上一篇的。 AnythingLLM是一个开源的全栈AI客户端,支持本地部署和API集成。它可以将任何文档或内容转化为上下文,供各种语言模型(LLM)在对话中使用。以下是…

信奥赛CSP-J复赛集训(模拟算法专题)(27):P5016 [NOIP 2018 普及组] 龙虎斗

信奥赛CSP-J复赛集训(模拟算法专题)(27):P5016 [NOIP 2018 普及组] 龙虎斗 题目背景 NOIP2018 普及组 T2 题目描述 轩轩和凯凯正在玩一款叫《龙虎斗》的游戏,游戏的棋盘是一条线段,线段上有 n n n 个兵营(自左至右编号 1 ∼ n 1 \sim n 1∼n),相邻编号的兵营之间…

多模态大模型常见问题

1.视觉编码器和 LLM 连接时,使用 BLIP2中 Q-Former那种复杂的 Adaptor 好还是 LLaVA中简单的 MLP 好,说说各自的优缺点? Q-Former(BLIP2): 优点:Q-Former 通过查询机制有效融合了视觉和语言特征…

SpringBoot项目实战(初级)

目录 一、数据库搭建 二、代码开发 1.pom.xml 2.thymeleaf模块处理的配置类 3.application配置文件 4.配置(在启动类中) 5.编写数据层 ②编写dao层 ③编写service层 接口 实现类 注意 补充(注入的3个注解) 1.AutoWir…

计算机网络——总结

01. 网络的发展及体系结构 网络演进历程 从1969年ARPANET的4个节点发展到如今覆盖全球的互联网,网络技术经历了电路交换到分组交换、有线连接到无线覆盖的革命性变革。5G时代的到来使得网络传输速度突破10Gbps,物联网设备数量突破百亿级别。 网络体系…

Umi-OCR- OCR 文字识别工具,支持截图、批量图片排版解析

Umi-OCR 是免费开源的离线 OCR 文字识别软件。无需联网,解压即用,支持截图、批量图片、PDF 扫描件的文字识别,能识别数学公式、二维码,可生成双层可搜索 PDF。内置多语言识别库,界面支持多语言切换,提供命令…

高速网络包处理,基础网络协议上内核态直接处理数据包,XDP技术的原理

文章目录 预备知识TCP/IP 网络模型(4层、7层)iptables/netfilterlinux网络为什么慢 DPDKXDPBFPeBPFXDPXDP 程序典型执行流通过网络协议栈的入包XDP 组成 使用 GO 编写 XDP 程序明确流程选择eBPF库编写eBPF代码编写Go代码动态更新黑名单 预备知识 TCP/IP…

C++:背包问题习题

1. 货币系统 1371. 货币系统 - AcWing题库 给定 V 种货币(单位:元),每种货币使用的次数不限。 不同种类的货币,面值可能是相同的。 现在,要你用这 V 种货币凑出 N 元钱,请问共有多少种不同的…

数据可信安全流通实战,隐语开源社区Meetup武汉站开放报名

隐语开源社区 Meetup 系列再出发!2025 年将以武汉为始发站,聚焦"技术赋能场景驱动",希望将先进技术深度融入数据要素流转的各个环节,推动其在实际应用场景中落地生根,助力释放数据要素的最大潜能&#xff01…

java使用Apache POI 操作word文档

项目背景: 当我们对一些word文档(该文档包含很多的标题比如 1.1 ,1.2 , 1.2.1.1, 1.2.2.3)当我们删除其中一项或者几项时,需要手动的对后续的进行补充。该功能主要是对标题进行自动的补充。 具…

免费开源的NAS解决方案:TrueNAS

TrueNAS是业内知名的FreeNAS系统的升级版,是一款开源的网络存储系统,具有高性能、稳定性和易用性等优点。 TrueNAS目前有三个版本,分别是TrueNAS CORE、TrueNAS ENTERPRISE、TrueNAS SCALE。其中,TrueNAS CORE基于FreeBSD开发&…