掌握 C# 设计模式:从基础到依赖注入
设计模式是一种可以在开发中重复使用的解决方案,能够提高代码的可维护性、扩展性和复用性。C# 中常见的设计模式包括单例模式、工厂模式、观察者模式、策略模式等。本文将介绍这些常见的设计模式,并探讨 SOLID 原则和依赖注入(Dependency Injection)的重要性。
1. 常见设计模式
单例模式(Singleton)
单例模式 确保一个类只有一个实例,并提供全局访问点。它在需要共享资源或控制全局状态的情况下非常有用。
public class Singleton
{private static Singleton instance;private static readonly object lockObj = new object();private Singleton() { }public static Singleton Instance{get{lock (lockObj){if (instance == null){instance = new Singleton();}return instance;}}}
}
在上面的例子中,Singleton 类通过静态 Instance 属性保证全局唯一性,并使用双重锁定确保线程安全。
工厂模式(Factory)
工厂模式 提供了一种创建对象的接口,而不是直接实例化类。它将对象的创建与业务逻辑解耦。
public interface IProduct
{void Create();
}public class ConcreteProductA : IProduct
{public void Create() => Console.WriteLine("Product A created");
}public class ConcreteProductB : IProduct
{public void Create() => Console.WriteLine("Product B created");
}public class ProductFactory
{public static IProduct GetProduct(string type){return type switch{"A" => new ConcreteProductA(),"B" => new ConcreteProductB(),_ => throw new ArgumentException("Invalid product type")};}
}
在这个示例中,工厂方法 GetProduct 负责创建 ConcreteProductA 或 ConcreteProductB,而客户端无需知道具体实现细节。
观察者模式(Observer)
观察者模式 定义了对象间的一对多依赖关系,当一个对象的状态改变时,依赖它的对象会收到通知并自动更新。这个模式常用于事件驱动的系统中。
public class Subject
{private List<IObserver> observers = new List<IObserver>();public void Attach(IObserver observer) => observers.Add(observer);public void Detach(IObserver observer) => observers.Remove(observer);public void Notify(){foreach (var observer in observers){observer.Update();}}
}public interface IObserver
{void Update();
}public class ConcreteObserver : IObserver
{public void Update() => Console.WriteLine("Observer notified");
}
在这个示例中,Subject 维护了一个观察者列表,状态改变时,Notify 方法通知所有观察者。
策略模式(Strategy)
策略模式 允许在运行时选择算法或策略,避免在代码中使用大量的条件语句。每个策略都是一个独立的类,封装了具体的算法或行为。
public interface IStrategy
{void Execute();
}public class ConcreteStrategyA : IStrategy
{public void Execute() => Console.WriteLine("Executing Strategy A");
}public class ConcreteStrategyB : IStrategy
{public void Execute() => Console.WriteLine("Executing Strategy B");
}public class Context
{private IStrategy strategy;public Context(IStrategy strategy) => this.strategy = strategy;public void ExecuteStrategy() => strategy.Execute();
}
在这个示例中,Context 类根据不同的策略执行相应的算法,策略可以在运行时动态选择。
2. SOLID 原则
SOLID 是面向对象设计的五大原则,旨在提高代码的可维护性和扩展性。这五个原则包括:
- S:单一职责原则(Single Responsibility Principle)
- 一个类应该只有一个职责。
- O:开闭原则(Open/Closed Principle)
- 软件实体应该对扩展开放,对修改关闭。
- L:里氏替换原则(Liskov Substitution Principle)
- 子类对象应该可以替换父类对象,并且不会导致错误。
- I:接口隔离原则(Interface Segregation Principle)
- 接口应该小而专,避免臃肿的接口。
- D:依赖倒置原则(Dependency Inversion Principle)
- 高层模块不应该依赖于低层模块,二者都应该依赖于抽象。
这些原则确保代码设计更加清晰、灵活,并且容易扩展。例如,遵循单一职责原则可以避免一个类承担太多职责,增加了代码的可维护性;而开闭原则确保我们可以在不修改现有代码的情况下进行功能扩展。
3. 依赖注入(Dependency Injection)
依赖注入 是实现 SOLID 原则(尤其是依赖倒置原则)的重要手段。它允许我们将类的依赖项通过外部注入,而不是让类自行实例化依赖项,从而提高代码的可测试性和灵活性。
构造函数注入
public interface IService
{void Serve();
}public class Service : IService
{public void Serve() => Console.WriteLine("Service Called");
}public class Client
{private readonly IService service;public Client(IService service){this.service = service;}public void Start() => service.Serve();
}
在这个示例中,Client 类不直接依赖 Service 类,而是依赖于 IService 接口,通过构造函数将实现传递给 Client,使得 Client 更加灵活。
使用依赖注入框架
在 C# 中,通常会使用依赖注入框架(如 ASP.NET Core 的内置 DI 容器)来自动管理依赖关系。
public void ConfigureServices(IServiceCollection services)
{services.AddTransient<IService, Service>();services.AddTransient<Client>();
}
通过 DI 容器,可以自动解析依赖项并注入到构造函数中,无需手动实例化对象。这种方式极大简化了对象的管理,提升了可维护性和可测试性。
结论
设计模式和 SOLID 原则是提高代码质量、灵活性和可扩展性的有效工具。通过合理运用设计模式,我们可以解决常见的开发问题,简化系统的设计。SOLID 原则确保我们的代码结构更加清晰,避免复杂的耦合和难以维护的代码。
- 单例模式:控制类的实例数量,适用于全局唯一对象。
- 工厂模式:通过工厂类创建对象,解耦了对象的创建和使用。
- 观察者模式:使得对象间的变化可以被自动通知,适用于事件驱动的场景。
- 策略模式:允许动态选择行为或算法,避免条件判断过多的代码。
- SOLID 原则 提高了代码设计的健壮性。
- 依赖注入:通过外部注入依赖项,减少了类之间的耦合。
掌握这些设计模式和原则可以显著提升开发效率和代码质量。如果你有更多关于设计模式或依赖注入的疑问,欢迎继续探讨!
这篇博客为你介绍了常见的设计模式、SOLID 原则以及依赖注入的相关概念。如果你有任何问题或需要具体的实现示例,欢迎联系我!
相关文章:
掌握 C# 设计模式:从基础到依赖注入
设计模式是一种可以在开发中重复使用的解决方案,能够提高代码的可维护性、扩展性和复用性。C# 中常见的设计模式包括单例模式、工厂模式、观察者模式、策略模式等。本文将介绍这些常见的设计模式,并探讨 SOLID 原则和依赖注入(Dependency Inj…...
根据json转HttpClient脚本
String json “{\n” " “paths”: {\n" " “/dev-api/system/subjectResult/exportUserList”: {\n" " “post”: {\n" " “tags”: [\n" " “bd-subject-result-controller”\n" " ],\n" " “summ…...
如何将LiDAR坐标系下的3D点投影到相机2D图像上
将激光雷达点云投影到相机图像上做数据层的前融合,或者把激光雷达坐标系下标注的物体点云的3d bbox投影到相机图像上画出来,都需要做点云3D点坐标到图像像素坐标的转换计算,也就是LiDAR 3D坐标转像素坐标。 看了网上一些文章都存在有错误或者…...
JAVA就业笔记6——第二阶段(3)
课程须知 A类知识:工作和面试常用,代码必须要手敲,需要掌握。 B类知识:面试会问道,工作不常用,代码不需要手敲,理解能正确表达即可。 C类知识:工作和面试不常用,代码不…...
02.04、分割链表
02.04、[中等] 分割链表 1、题目描述 给你一个链表的头节点 head 和一个特定值 x ,请你对链表进行分隔,使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。 你不需要 保留 每个分区中各节点的初始相对位置。 2、解题思路 本题要求将链表分隔…...
Excel 中根据患者的就诊时间标记病例为“初诊”或“复诊”
1. 假设: 患者表:包含患者的基本信息,如患者 ID 和患者姓名。 病例表:包含病例信息,如患者 ID、就诊时间和就诊状态。 2. 操作步骤: 合并数据: 确保病例表中有一列包含患者 ID,以…...
遇到“mfc100u.dll丢失”的系统错误要怎么处理?科学修复mfc100u.dll
遇到“mfc100u.dll丢失”的系统错误会非常麻烦,因为mfc100u.dll是Microsoft Visual C 2010 Redistributable Package的重要部分,许多应用程序和游戏在运行时都需要调用这个文件。如果这个文件缺失,可能会导致相关软件或游戏启动失败。面对这种…...
[Linux] 逐层深入理解文件系统 (1)—— 进程操作文件
标题:[Linux] 文件系统 (1)—— 进程操作文件 个人主页水墨不写bug (图片来源于网络) 目录 一、进程与打开的文件 二、文件的系统调用与库函数的关系 1.系统调用open() 三、内存中的文件描述符表 四、缓冲区…...
RT-Thread 互斥量的概念
目录 概述 1 互斥量定义 1.1 概念介绍 1.2 线程优先级翻转问题 2 互斥量管理 2.1 结构体定义 2.2 函数接口介绍 2.2.1 rt_mutex_create函数 2.2.2 rt_mutex_delete 函数 2.2.3 初始化和脱离互斥量 概述 本文主要介绍互斥量的概念,实现原理。还介绍RT-Thre…...
6.计算机网络_UDP
UDP的主要特点: 无连接,发送数据之前不需要建立连接。不保证可靠交付。面向报文。应用层给UDP报文后,UDP并不会抽象为一个一个的字节,而是整个报文一起发送。没有拥塞控制。网络拥堵时,发送端并不会降低发送速率。可以…...
Windows应急响蓝安服面试
Windows应急响应 蓝队溯源流程 学习Windows应急首先要站在攻击者的角度去学习一些权限维持和权限提升的方法.,文章中的方法其实和内网攻防笔记有类似l红队教你怎么利用 蓝队教你怎么排查 攻防一体,应急响应排查这些项目就可以 端口/服务/进程/后门文件都是为了权限维持,得到s…...
PCL 点云配准-4PCS算法(粗配准)
目录 一、概述 1.1原理 1.2实现步骤 1.3应用场景 二、代码实现 2.1关键函数 2.1.1 加载点云数据 2.1.2 执行4PCS粗配准 2.1.3 可视化源点云、目标点云和配准结果 2.2完整代码 三、实现效果 3.1原始点云 3.2配准后点云 PCL点云算法汇总及实战案例汇总的目录地址链接…...
12、论文阅读:利用生成对抗网络实现无监督深度图像增强
Towards Unsupervised Deep Image Enhancement With Generative Adversarial Network 摘要介绍相关工作传统图像增强基于学习的图像增强 论文中提出的方法动机和目标网络架构损失函数1) 质量损失2) 保真损失3)身份损失4)Total Loss 实验 摘要 提高图像的…...
Axure重要元件三——中继器表单制作
亲爱的小伙伴,在您浏览之前,烦请关注一下,在此深表感谢! 本节课:中继器表单制作 课程内容:利用中继器制作表单 应用场景:台账、表单 案例展示: 步骤一:建立一个背景区…...
DMAIC赋能智能家居:解锁未来生活新篇章!
从清晨自动拉开的窗帘,到夜晚自动调暗的灯光,每一处细节都透露着科技的温度与智慧的光芒。而在这场智能革命的浪潮中,DMAIC(定义Define、测量Measure、分析Analyze、改进Improve、控制Control)作为六西格玛管理的核心方…...
代码随想录算法训练营第二天| 209.长度最小的子数组 59.螺旋矩阵II 区间和 开发商购买土地
209. 长度最小的子数组 题目: 给定一个包含正整数的数组 nums 和一个正整数 target ,找出该数组中满足其和 ≥ target 的长度最小的 连续子数组 ,并返回其长度。如果不存在符合条件的子数组,返回 0。 示例: 示例 1…...
mysql隐藏索引
1. 什么是隐藏索引? 在 MySQL 8 中,隐藏索引(Invisible Indexes)是指一种特殊类型的索引,它并不真正被删除,而是被标记为“不可见”。当索引被标记为不可见时,查询优化器在生成查询计划时将忽略…...
etcd入门到实战
概述:本文将介绍etcd特性、使用场景、基本原理以及Linux环境下的实战操作 入门 什么是etcd? etcd是一个分布式键值存储数据库 关键字解析: 键值存储:存储协议是 key—value 的形式,类似于redis分布式:…...
Build an Android project and get a `.apk` file on a Debian 11 command line
You can build an Android project and get a .apk file on a Debian 11 command line without using Android Studio. The process involves using the Android SDK command-line tools (sdkmanager, adb, and gradle). Here’s a step-by-step guide to building the ???…...
解读 Java 经典巨著《Effective Java》90条编程法则,第4条:通过私有构造器强化不可实例化的能力
文章目录 【前言】欢迎订阅【解读《Effective Java》】系列专栏java.lang.Math 类的设计经验总结 【前言】欢迎订阅【解读《Effective Java》】系列专栏 《Effective Java》是 Java 开发领域的经典著作,作者 Joshua Bloch 以丰富的经验和深入的知识,全面…...
以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:
一、属性动画概述NETX 作用:实现组件通用属性的渐变过渡效果,提升用户体验。支持属性:width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项: 布局类属性(如宽高)变化时&#…...
前端倒计时误差!
提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...
土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测;从基础到高级,涵盖ArcGIS数据处理、ENVI遥感解译与CLUE模型情景模拟等
🔍 土地利用/土地覆盖数据是生态、环境和气象等诸多领域模型的关键输入参数。通过遥感影像解译技术,可以精准获取历史或当前任何一个区域的土地利用/土地覆盖情况。这些数据不仅能够用于评估区域生态环境的变化趋势,还能有效评价重大生态工程…...
C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)
名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...
Unity UGUI Button事件流程
场景结构 测试代码 public class TestBtn : MonoBehaviour {void Start(){var btn GetComponent<Button>();btn.onClick.AddListener(OnClick);}private void OnClick(){Debug.Log("666");}}当添加事件时 // 实例化一个ButtonClickedEvent的事件 [Formerl…...
MySQL:分区的基本使用
目录 一、什么是分区二、有什么作用三、分类四、创建分区五、删除分区 一、什么是分区 MySQL 分区(Partitioning)是一种将单张表的数据逻辑上拆分成多个物理部分的技术。这些物理部分(分区)可以独立存储、管理和优化,…...
Golang——7、包与接口详解
包与接口详解 1、Golang包详解1.1、Golang中包的定义和介绍1.2、Golang包管理工具go mod1.3、Golang中自定义包1.4、Golang中使用第三包1.5、init函数 2、接口详解2.1、接口的定义2.2、空接口2.3、类型断言2.4、结构体值接收者和指针接收者实现接口的区别2.5、一个结构体实现多…...
通过MicroSip配置自己的freeswitch服务器进行调试记录
之前用docker安装的freeswitch的,启动是正常的, 但用下面的Microsip连接不上 主要原因有可能一下几个 1、通过下面命令可以看 [rootlocalhost default]# docker exec -it freeswitch fs_cli -x "sofia status profile internal"Name …...
Vue 模板语句的数据来源
🧩 Vue 模板语句的数据来源:全方位解析 Vue 模板(<template> 部分)中的表达式、指令绑定(如 v-bind, v-on)和插值({{ }})都在一个特定的作用域内求值。这个作用域由当前 组件…...
