C# 设计模式的六大原则(SOLID)
C# 设计模式的六大原则(SOLID)
引言
在面向对象编程中,设计模式提供了高效、可复用和可维护的代码结构。SOLID原则是软件设计中的一组重要原则,用于确保代码具有良好的可维护性、可扩展性和灵活性。SOLID是五个设计原则的首字母缩写,广泛应用于面向对象编程中,尤其是在大型项目中。
本文将详细介绍SOLID五大设计原则以及它们在C#中的应用,并进一步探讨有时被扩展为“六大原则”的情况。
1. 单一职责原则(SRP)
定义:
单一职责原则(Single Responsibility Principle)规定一个类应该只有一个职责,并且该类应该只有一个引起它变化的原因。也就是说,一个类不应该承担过多的责任,否则会变得难以维护和扩展。
在C#中的实现:
public class OrderProcessor
{public void ProcessOrder(Order order){// 处理订单逻辑}
}public class OrderPrinter
{public void PrintOrder(Order order){// 打印订单逻辑}
}
解释:
在这个例子中,OrderProcessor 类只负责处理订单,而 OrderPrinter 类只负责打印订单。通过分离不同的职责,使得每个类的功能更加专一、简洁。
优势:
- 提高可维护性:更改一个类的逻辑时,不会影响其他无关的部分。
- 增强可测试性:由于职责明确,单元测试变得更加简单。
2. 开闭原则(OCP)
定义:
开闭原则(Open/Closed Principle)表明,“软件实体(类、模块、函数等)应该对扩展开放,对修改封闭”。这意味着当需求变化时,我们应该通过添加新代码来扩展系统,而不是修改现有代码。
在C#中的实现:
public interface IPaymentMethod
{void Pay();
}public class CreditCardPayment : IPaymentMethod
{public void Pay() { /* 支付实现 */ }
}public class PayPalPayment : IPaymentMethod
{public void Pay() { /* 支付实现 */ }
}public class PaymentProcessor
{public void ProcessPayment(IPaymentMethod paymentMethod){paymentMethod.Pay();}
}
解释:
通过使用接口(IPaymentMethod),我们可以为不同的支付方式实现扩展。增加新的支付方式时,我们只需创建新的实现类,而不需要修改 PaymentProcessor 类。
优势:
- 灵活性:新功能可以通过增加新类而不破坏现有系统的稳定性。
- 增强可扩展性:能够适应系统需求的变化,而无需修改已有的代码。
3. 里氏替换原则(LSP)
定义:
里氏替换原则(Liskov Substitution Principle)规定,子类对象应该能够替换父类对象,并且不会改变程序的正确性。也就是说,子类必须遵循父类的行为约定,并可以在任何父类对象出现的地方替代父类。
在C#中的实现:
public class Bird
{public virtual void Fly() { /* 通用飞行实现 */ }
}public class Sparrow : Bird
{public override void Fly() { /* 麻雀飞行实现 */ }
}public class Penguin : Bird
{public override void Fly(){throw new NotSupportedException("企鹅不能飞");}
}
解释:
在上面的代码中,Penguin 类违反了里氏替换原则,因为企鹅不能飞,强制让其实现 Fly 方法不符合实际需求。我们可以通过接口或其他设计方式来避免这种问题。
优势:
- 增强代码的可替换性:子类应该能无缝替换父类,确保系统的健壮性。
- 减少代码错误:遵循该原则避免不必要的设计冲突。
4. 接口隔离原则(ISP)
定义:
接口隔离原则(Interface Segregation Principle)要求类应该仅实现它需要使用的接口,而不应该强迫它实现不需要的方法。换句话说,接口应该细化,而不是做一个“臃肿”的接口。
在C#中的实现:
public interface IPrinter
{void Print();
}public interface IFax
{void Fax();
}public class MultiFunctionMachine : IPrinter, IFax
{public void Print() { /* 打印实现 */ }public void Fax() { /* 传真实现 */ }
}public class Printer : IPrinter
{public void Print() { /* 打印实现 */ }
}
解释:
在这个例子中,Printer 类只实现了与打印相关的接口,而 MultiFunctionMachine 类同时实现了打印和传真接口。根据接口隔离原则,如果一个类只关心打印,那么它不应该实现传真接口。
优势:
- 减少不必要的依赖:减少类与类之间的不必要耦合。
- 增强代码灵活性:提高代码的复用性,减少修改的影响范围。
5. 依赖倒转原则(DIP)
定义:
依赖倒转原则(Dependency Inversion Principle)要求高层模块不应该依赖低层模块,而应该依赖抽象。具体来说,应该依赖接口或抽象类,而不是具体的实现类。
在C#中的实现:
public interface IDatabase
{void SaveData();
}public class SQLServerDatabase : IDatabase
{public void SaveData() { /* SQL Server 数据库实现 */ }
}public class BusinessLogic
{private readonly IDatabase _database;public BusinessLogic(IDatabase database){_database = database;}public void ProcessData() { _database.SaveData(); }
}
解释:
在这个例子中,BusinessLogic 类依赖于 IDatabase 接口,而不是具体的 SQLServerDatabase 类。通过这种方式,我们可以轻松地将不同的数据库实现替换到 BusinessLogic 中。
优势:
- 提高模块的灵活性:减少模块之间的耦合,增强可扩展性。
- 便于单元测试:可以使用模拟对象(mock)轻松替换数据库实现,进行单元测试。
6. 总结
SOLID原则是现代面向对象设计中的基石,帮助开发人员编写高效、可维护、可扩展的代码。通过遵循这些原则,可以极大提高系统的质量、可读性和可测试性。在实际开发中,理解和应用SOLID原则对于构建稳定和高质量的应用程序至关重要。
希望这篇博客能够帮助你更好地理解并应用SOLID原则。如果你有任何问题,欢迎在评论区讨论或者留言交流!
相关文章:
C# 设计模式的六大原则(SOLID)
C# 设计模式的六大原则(SOLID) 引言 在面向对象编程中,设计模式提供了高效、可复用和可维护的代码结构。SOLID原则是软件设计中的一组重要原则,用于确保代码具有良好的可维护性、可扩展性和灵活性。SOLID是五个设计原则的首字母…...
数据库自增 id 过大导致前端时数据丢失
可以看到,前端响应参数是没有丢失精度的 但是在接受 axios 请求参数时出现了精度丢失 解决方案一:改变 axios 字符编码 axios.defaults.headers[Content-Type] application/json;charsetUTF-8; 未解决 解决方案二:手动使用 json.parse() …...
第二十六天 自然语言处理(NLP)词嵌入(Word2Vec、GloVe)
自然语言处理(NLP)中的词嵌入(Word2Vec、GloVe)技术,是NLP领域的重要组成部分,它们为词汇提供了高维空间到低维向量的映射,使得语义相似的词汇在向量空间中的距离更近。以下是对这些技术的详细解…...
MongoDB 固定集合
MongoDB 固定集合 MongoDB中的固定集合(Capped Collections)是一种具有固定大小的集合,当集合中的数据达到其最大大小时,它会自动覆盖最早的文档。这种类型的集合在MongoDB中用于实现高效的、固定大小的循环缓冲区。本文将详细介…...
数据结构9.3 - 文件基础(C++)
目录 1 打开文件字符读写关闭文件 上图源自:https://blog.csdn.net/LG1259156776/article/details/47035583 1 打开文件 法 1法 2ofstream file(path);ofstream file;file.open(path); #include<bits/stdc.h> using namespace std;int main() {char path[]…...
Leetcode 1254 Number of Closed Islands + Leetcode 1020 Number of Enclaves
Leetcode 1254 题意 给定一个m*n的矩阵含有0和1,1代表水,0代表陆地,岛屿是陆地的集合,如果一个岛屿和四个方向的边界相连,则不算封闭岛屿。求有多少个封闭的岛屿。 题目链接 https://leetcode.com/problems/number…...
Junit4单元测试快速上手
文章目录 POM依赖引入业务层测试代码Web层测试代码生成测试类文件 在工作中我用的最多的单元测试框架是Junit4。通常在写DAO、Service、Web层代码的时候都会进行单元测试,方便后续编码,前端甩锅。 POM依赖引入 <dependency><groupId>org.spr…...
U盘提示格式化?原因、恢复方案与预防措施全解析
一、U盘提示格式化现象概述 在日常使用U盘的过程中,我们有时会遇到一个令人头疼的问题——U盘插入电脑后,系统却弹出一个提示框,告知我们U盘需要格式化才能访问。这个提示往往伴随着数据的潜在丢失风险,让我们不禁为之心焦。U盘提…...
HTML——13.超链接
<!DOCTYPE html> <html><head><meta charset"UTF-8"><title>超链接</title></head><body><!--超链接:从一个网页链接到另一个网页--><!--语法:<a href"淘宝网链接的地址"> 淘宝…...
vue中的设计模式
vue中使用了哪些设计模式 1. 观察者模式(Observer Pattern) 应用场景:Vue 的响应式系统核心就是观察者模式。 实现方式:通过 Object.defineProperty 或 Proxy 监听数据变化,当数据发生变化时,通知依赖的视…...
利用python将图片转换为pdf格式的多种方法,实现批量转换,内置模板代码,全网最全,超详细!!!
文章目录 前言1、img2pdf库的使用1.1 安装img2pdf库1.2 案例演示(模板代码) 2、Pillow库的使用2.1 pillow库的安装2.2 案例演示(模板代码) 3、PyMuPDF库的使用3.1 安装pymupdf库3.2 案例演示(模板代码)3.3 …...
tcpdump的常见方法
详解tcpdump的使用方法:网络数据包捕获与分析 tcpdump是一个功能强大的命令行工具,用于捕获和分析通过网络接口传输的数据包。它广泛应用于网络故障诊断、网络安全监控和协议分析等领域。本文将详细介绍tcpdump的使用方法,包括安装、基本命令…...
工控主板ESM7000/6800E支持远程桌面控制
英创公司ESM7000 是面向工业领域的双核 Cortex-A7 高性能嵌入式主板,ESM6800E则为单核Cortex-A7 高性价比嵌入式主板,ESM7000、ESM6800E都是公司的成熟产品,已广泛应用于工业很多领域。ESM7000/6800E板卡中Linux系统配置为linux-4.9.11内核、…...
wamp php7.4 运行dm8
背景 1、电脑安装了dm8,具体参照官网dm8安装 2、安装好了wamp,我当前的php版本切换成了7.4的,我wamp的安装路径d:\wamp64\ 操作 3、查看phpinfo,如果Thread Safet为enabled,则选择pdo74_dm.dll,否则选择…...
HTML5 进度条(Progress Bar)详解
HTML5 进度条(Progress Bar)详解 进度条是用于显示任务完成进度的控件,常用于加载、上传或下载等操作。HTML5提供了原生的<progress>元素,使得创建进度条变得简单和直观。 1. 基本用法 <progress>元素的基本语法如…...
LabVIEW开发中常见硬件通讯接口快速识别
在 LabVIEW 开发中,与硬件进行通讯是实现数据采集与控制的重要环节。准确判断通讯接口类型和协议,可以提高开发效率,减少调试时间。本文结合 LabVIEW 的实际应用,详细介绍如何识别和判断常见硬件通讯接口的定义,并提供…...
高频 SQL 50 题(基础版)_1068. 产品销售分析 I
销售表 Sales: (sale_id, year) 是销售表 Sales 的主键(具有唯一值的列的组合)。 product_id 是关联到产品表 Product 的外键(reference 列)。 该表的每一行显示 product_id 在某一年的销售情况。 注意: price 表示每…...
笔记:一次mysql主从复制延迟高的处理尝试
背景 mysql 5.7 主从复制 主库进行了一次灌数,导致多个大事务产生,主从延迟下不去,经确认该表最终truncate,并且该表仅有insert和select操作,故对该表的事务进行跳过,直到同步至truncate 跳过事务需谨慎&…...
基于C语言的卡丁车管理系统【控制台应用程序】
注意:需要提前创建对应的.dat文件 本项目实现了数据的永久存储,有用户的注册、登录。 管理员对卡丁车的管理、查看预约用户、修改帐户权限。 用户对个人信息的管理、查看并预约卡丁车、卡丁车维修上报。 维修员对卡丁车的维修状态上报、个人信息管理。 …...
Docker 搭建 Gogs
Docker 搭建 Gogs 准备工作 先准备配置目录和持久化目录,举个栗子:mkdir -p /opt/module/gogs/{data,backup} 拉取官方Gogs镜像 # 拉取 gogs:0.13 docker pull gogs/gogs:0.13 # 拉取最新版 gogs 镜像 docker pull gogs/gogs启动一个临时容器【通过创…...
变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析
一、变量声明设计:let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性,这种设计体现了语言的核心哲学。以下是深度解析: 1.1 设计理念剖析 安全优先原则:默认不可变强制开发者明确声明意图 let x 5; …...
业务系统对接大模型的基础方案:架构设计与关键步骤
业务系统对接大模型:架构设计与关键步骤 在当今数字化转型的浪潮中,大语言模型(LLM)已成为企业提升业务效率和创新能力的关键技术之一。将大模型集成到业务系统中,不仅可以优化用户体验,还能为业务决策提供…...
蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练
前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1):从基础到实战的深度解析-CSDN博客,但实际面试中,企业更关注候选人对复杂场景的应对能力(如多设备并发扫描、低功耗与高发现率的平衡)和前沿技术的…...
在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用
1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...
JVM虚拟机:内存结构、垃圾回收、性能优化
1、JVM虚拟机的简介 Java 虚拟机(Java Virtual Machine 简称:JVM)是运行所有 Java 程序的抽象计算机,是 Java 语言的运行环境,实现了 Java 程序的跨平台特性。JVM 屏蔽了与具体操作系统平台相关的信息,使得 Java 程序只需生成在 JVM 上运行的目标代码(字节码),就可以…...
Mysql中select查询语句的执行过程
目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析(Parser) 2.4、执行sql 1. 预处理(Preprocessor) 2. 查询优化器(Optimizer) 3. 执行器…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...
LabVIEW双光子成像系统技术
双光子成像技术的核心特性 双光子成像通过双低能量光子协同激发机制,展现出显著的技术优势: 深层组织穿透能力:适用于活体组织深度成像 高分辨率观测性能:满足微观结构的精细研究需求 低光毒性特点:减少对样本的损伤…...
【 java 虚拟机知识 第一篇 】
目录 1.内存模型 1.1.JVM内存模型的介绍 1.2.堆和栈的区别 1.3.栈的存储细节 1.4.堆的部分 1.5.程序计数器的作用 1.6.方法区的内容 1.7.字符串池 1.8.引用类型 1.9.内存泄漏与内存溢出 1.10.会出现内存溢出的结构 1.内存模型 1.1.JVM内存模型的介绍 内存模型主要分…...
十九、【用户管理与权限 - 篇一】后端基础:用户列表与角色模型的初步构建
【用户管理与权限 - 篇一】后端基础:用户列表与角色模型的初步构建 前言准备工作第一部分:回顾 Django 内置的 `User` 模型第二部分:设计并创建 `Role` 和 `UserProfile` 模型第三部分:创建 Serializers第四部分:创建 ViewSets第五部分:注册 API 路由第六部分:后端初步测…...
