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

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.rhkb.cn/news/38180.html

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

相关文章

asp.net core mvc模块化开发

razor类库 新建PluginController using Microsoft.AspNetCore.Mvc;namespace RazorClassLibrary1.Controllers {public class PluginController : Controller{public IActionResult Index(){return View();}} }Views下Plugin下新建Index.cshtml {ViewBag.Title "插件页…

边缘计算革命:重构软件架构的范式与未来

摘要 边缘计算通过将算力下沉至网络边缘,正在颠覆传统中心化软件架构的设计逻辑。本文系统分析了边缘计算对软件架构的范式革新,包括分布式分层架构、实时资源调度、安全防护体系等技术变革,并结合工业物联网、智慧医疗等场景案例&#xff0c…

单链表:数据结构的灵动之链

本文主要讲解链表的概念和结构以及实现单链表 目录 一、链表的概念及结构 二、单链表的实现 1.1链表的实现: 1.2单链表的实现: 单链表尾插: 单链表的头插: 单链表的尾删: 单链表头删: 单链表查找&#…

链表题型-链表操作-JS

一定要注意链表现在的头节点是空节点还是有值的节点。 一、移除链表中的元素 有两种方式,直接使用原来的链表进行删除操作;设置一个虚拟头节点进行删除操作。 直接使用原来的链表进行删除操作时,需要考虑是不是头节点,因为移除…

读《浪潮之巅》:探寻科技产业的兴衰密码

引言:邂逅《浪潮之巅》 在信息技术飞速发展的今天,科技公司如繁星般闪烁,又似流星般划过。而我与《浪潮之巅》的相遇,就像在浩渺的科技海洋中,发现了一座指引方向的灯塔。初次听闻这本书,是在一次技术交流会…

【和春笋一起学C++】文本文件I/O

在windows系统中读取键盘的输入和在屏幕上显示输出统称为:控制台输入/输出。把读取文本文件和把字符输出到文本文件中统称为:文本文件I/O。 目录 1. 输出文本文件 2. 读取文本文件 1. 输出文本文件 把字符输出到文本文件中和输出到控制台很相似&#x…

【C#】WinForm自定义控件及窗体

前言 WinForm(Windows Forms)是Microsoft.NET框架中的技术,用于开发Windows桌面应用程序。它提供了一套丰富的控件和组件。通过拖放控件、编写事件处理程序等方式快速构建用户界面。 通过属性窗口定制这些控件的外观和行为。 通过数据绑定&am…

Live555+Windows+MSys2 编译Androidso库和运行使用

下载 wget http://www.live555.com/liveMedia/public/live555-latest.tar.gz tar -xzvf live555-latest.tar.gz加入版本控制 git init git add . git commit -a -m "first init" git log修改config.android-arm64 cd live vim config.android-arm64 ./genMakefile…

大模型-提示词工程与架构

什么是提示工程 提示工程(Prompt Engineering)是一门新兴的技术领域,专注于研究如何设计、构建和优化提示词,以充分发挥大模型的潜力 。它涉及到对语言结构、任务需求、模型特性等多方面因素的综合考量。提示工程的目标是通过精心…

Agent Team 多智能体系统解析

引言 在人工智能技术高速发展的今天,"多智能体协作系统"(Agent Team)正成为突破效率瓶颈的关键技术。与传统的单体AI不同,这种由多个专业化智能体组成的协同网络,通过分工协作和动态调整,展现出…

【蓝桥杯—单片机】IAP15F2K61S2专项 | 真题整理、解析与拓展 | 省赛题(更新ing...)

IAP15F2K61S2 专项 前言IAP15F2K61S2 介绍(基于手册)I/O口结构复位管脚RST中断第十四届省赛 外设通过PWM控制第十五届省赛题 性能与工作参数在线调试第十四届省赛题拓展与小结:单片机在线调试常用的接口 功耗第十五届省赛题 前言 在本文中我…

生物化学笔记:医学免疫学原理02 抗原概念+免疫应答+抗原的分类

抗原基本概念 影响抗原刺激机体产生免疫应答的因素 抗原的分类 CG 【北京大学】1080p 王月丹教授 《医学免疫学原理》2022春 全81p

(UI自动化测试)第二篇:元素定位的方法_name定位

二、name定位 ⽅法: driver.find_element_by_name(“name属性值”) 前置: 标签必须name属性 特点: 当前⻚⾯可以重复 提示: 由于name属性值可以重复,所以使⽤时需要查看是否为唯⼀。 # 导包selenium from selenium i…

软考中级-软件设计师 准备

软考中级-软件设计师 准备 一、软考相关1.1、考试时间1.2、考试时长1.3、题型和分值: 二、软考备考2.1、相关书籍2.2、推荐课程:B站up主zst_20012.3、学习路线 一、软考相关 1.1、考试时间 一年有两次软考,一般是五月末和十一月的中旬 以下…

记忆力训练day24

一 数字锁链串联法 数字两位 两位的连

田间机器人幼苗视觉检测与护苗施肥装置研究(大纲)

田间机器人幼苗视觉检测与护苗施肥装置研究 基于多光谱视觉与精准施肥的农业机器人系统设计 第一章 绪论 1.1 研究背景与意义 农业智能化需求: 传统幼苗检测依赖人工,效率低且易遗漏弱苗/病苗施肥不精准导致资源浪费和环境污染 技术挑战:…

Debian12生产环境配置笔记

在 Debian 12 上进行生产环境配置的详细步骤,涵盖软件更新、基础软件安装、Docker 及 Redis 部署,以及 Nginx 配置多个虚拟主机等内容。所有命令均以 root 用户身份执行,无需添加 sudo 1. 更新软件 首先,确保系统上的所有软件包…

HAL库编程知识点---Can.c和Driver_can.c分层开发

在一个工程中,通常会把对CAN外设的操作分成底层和上层两个部分,从而提高代码的模块化和可维护性。一般来说: can.c 通常由硬件抽象层(HAL)或者自动生成工具(如 CubeMX)提供或生成。主要负责CAN硬…

7. 【Vue实战--孢子记账--Web 版开发】-- 收支分类设置

本篇文章我们一起来实现收支分类功能。收支分类和前篇文章的主币种设置界面大体类似。我们将详细介绍如何创建和管理不同的收支分类,以便用户可以更好地组织和跟踪他们的财务状况。 一、功能 先来看一下原型界面,界面很简单,这里就不多讲解…

人工智能 - DeepSeek 和 Manus 的区别和应用场景

DeepSeek 与 Manus 是人工智能领域两种不同技术路线的代表,其核心区别在于功能定位和技术实现,应用场景也因此存在显著差异。以下是两者的对比分析: 一、核心区别 技术定位 DeepSeek:定位为“超级大脑”,专注于底层大模型的研发,擅长处理数学题、代码生成、知识问答等需要…