C# 设计模式(创建型模式):单例模式

C# 设计模式(创建型模式):单例模式

1. 引言

在软件开发中,设计模式是解决常见问题的经典方法。单例模式(Singleton Pattern)是创建型设计模式中的一种,旨在确保某个类只有一个实例,并提供一个全局的访问点。在许多情况下,我们需要在应用程序中确保某个对象的唯一性,单例模式能够很好地满足这一需求。

本文将深入解析单例模式在 C# 中的实现方式,以及在实际开发中的应用。

2. 单例模式的定义

单例模式的核心思想是:确保某个类只有一个实例,并提供一个全局访问点。这个实例会在整个应用程序的生命周期内保持唯一,避免了多次创建同一个对象,减少了内存开销并保证了全局状态的一致性。

3. 单例模式的应用场景

单例模式通常适用于以下情况:

  • 全局访问点:我们需要全局唯一的资源,例如数据库连接池、配置管理、日志记录器等。
  • 控制资源的创建:避免频繁创建和销毁对象,减少系统开销。
  • 状态共享:对象的状态在不同组件之间共享。

4. 单例模式的实现

在 C# 中,单例模式有多种实现方式,以下是最常用的几种。

4.1. 懒汉式(Lazy Initialization)

懒汉式实现意味着在第一次访问实例时,才会创建该实例。这种方式可以节省内存,但需要确保线程安全。

public class Singleton
{private static Singleton _instance;private static readonly object _lock = new object();private Singleton() { }public static Singleton Instance{get{lock (_lock){if (_instance == null){_instance = new Singleton();}}return _instance;}}
}

优点

  • 在首次调用时才创建实例,节省内存。
  • 通过锁定机制保证线程安全。

缺点

  • 加锁的开销较大,在高并发场景下可能影响性能。
4.2. 饿汉式(Eager Initialization)

饿汉式实现是在类加载时就创建实例。这种方式简单,但不适用于实例化开销较大的场景,因为即使实例从未被使用,它也会被创建。

public class Singleton
{private static readonly Singleton _instance = new Singleton();private Singleton() { }public static Singleton Instance => _instance;
}

优点

  • 实现简单,线程安全。
  • 类加载时就初始化实例,不会受到多线程问题影响。

缺点

  • 无法控制实例化的时机,即使从未使用,也会在程序启动时创建实例,可能导致不必要的资源消耗。
4.3. 双重锁检查(Double-Check Locking)

为了提高性能,可以使用双重锁检查方式,在多线程环境下减少锁定的开销。

public class Singleton
{private static volatile Singleton _instance;private static readonly object _lock = new object();private Singleton() { }public static Singleton Instance{get{if (_instance == null){lock (_lock){if (_instance == null){_instance = new Singleton();}}}return _instance;}}
}

优点

  • 在多线程环境下具有较高的性能。
  • 只有第一次访问时才会加锁。

缺点

  • 比较复杂,容易出现同步问题,需要仔细处理。
4.4. 静态构造器方式

C# 中的静态构造器会在类第一次被访问时自动执行,因此也可以用它来实现单例模式。

public class Singleton
{private static readonly Singleton _instance = new Singleton();private Singleton() { }public static Singleton Instance => _instance;
}

优点

  • 简洁、线程安全。
  • 不需要显式加锁,CLR会保证静态构造器在多线程环境下的安全性。

缺点

  • 实例创建时机是由 CLR 控制,不能完全手动控制。

5. 单例模式的线程安全问题

单例模式在多线程环境下可能会遇到线程安全问题,特别是在懒汉式实现中。当多个线程同时访问实例时,可能导致多个实例被创建。为了解决这个问题,可以使用锁定机制(如 lock 关键字)来保证线程安全。

  • 锁定机制:保证在同一时刻只有一个线程能够创建实例。
  • volatile 关键字:确保实例的读取是最新的。

6. 单例模式的优缺点

优点:
  • 节省内存:避免了多次实例化对象,减少了内存开销。
  • 全局访问:提供全局唯一实例,便于访问和共享数据。
  • 延迟加载:可以在第一次访问时才创建实例,避免不必要的资源消耗。
缺点:
  • 全局状态:单例模式引入了全局状态,可能会导致系统难以维护和扩展。
  • 隐藏依赖:单例模式可能会导致类之间的隐式依赖,增加了代码的耦合度。

7. 总结

单例模式作为创建型设计模式中的一种,广泛应用于需要唯一实例的场景中。通过懒汉式、饿汉式、双重锁检查等方式,可以根据具体的需求选择最合适的实现方式。然而,在使用单例模式时,需要谨慎考虑其对系统可维护性和扩展性的影响。在某些情况下,过度使用单例模式可能导致系统设计不清晰,因此要权衡利弊,合理使用。

希望本文能够帮助你理解单例模式的核心思想和在 C# 中的实现方式。如果你有任何问题或改进建议,欢迎留言交流!


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

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

相关文章

系统思考—信任

《基业长青》作者指出:“在人生的重要十字路口,选择信任是一场赌注。信任带来的好处可能巨大,而失去信任的代价却相对有限。但如果选择不信任,最优秀的人才可能因失望而离开。” 在企业管理中,信任不仅是人际关系的纽…

推理加速:投机采样经典方法

一 SpecInfer 基于模型 SpecInfer([2305.09781] SpecInfer: Accelerating Generative Large Language Model Serving with Tree-based Speculative Inference and Verification) SpecInfer 投机采样利用多个小型模型(SSM)快速生…

深入理解Java中的Set集合:特性、用法与常见操作指南

一、HashSet集合 1.HashSet集合的特点 2.HashSet常用方法 ①:add(Object o):向Set集合中添加元素,不允许添加重复数据。 ②:size():返回Set集合中的元素个数 ③.remove(Object o): 删除Set集合中的obj对…

黑马Java面试教程_P10_设计模式

系列博客目录 文章目录 系列博客目录前言1. 工厂方法模式1.1 概述1.2 简单工厂模式1.2.1 结构1.2.2 实现1.2.3 优缺点 1.3 工厂方法模式1.3.1 概念1.3.2 结构1.3.3 实现1.3.4 优缺点 1.4 抽象工厂模式1.4.1 概念1.4.2 结构1.4.3 实现1.4.4 优缺点1.4.5 使用场景 总结&#xff0…

开源架构的容器化部署优化版

上三篇文章推荐: 开源架构的微服务架构实践优化版(New) 开源架构中的数据库选择优化版(New) 开源架构学习指南:文档与资源的智慧锦囊(New) 我管理的社区推荐:【青云交社区…

SpringCloudAlibaba实战入门之Sentinel服务降级和服务熔断(十五)

一、Sentinel概述 1、Sentinel是什么 随着微服务的流行,服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点,从流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性。 一句话概括:sentinel即Hystrix的替代品,官网: https://sentinelguard.io/zh…

【每日学点鸿蒙知识】导入cardEmulation、自定义装饰器、CallState状态码顺序、kv配置、签名文件配置

1、HarmonyOS 无法导入cardEmulation? 在工程entry mudule里的index.ets文件里导入cardEmulation失败 可以按照下面方式添加SystemCapability;在src/main/syscap.json(此文件需要手动创建)中添加如下内容 {"devices": {"gen…

Datawhale AI冬令营(第二期)动手学AI Agent--Task3:学Agent工作流搭建,创作进阶Agent

目录 一、工作流:制作复杂Agent的福音! 二、支付宝百宝箱中工作流介绍 三、设计工作流 3.1 准备功能模块 3.2组合工作流 3.3 模块测试需要注意什么 3.4迭代优化 四、高中学习小助手工作流设计 4.1 选题调研 4.2 功能模块设计 4.3 组合完整工作…

Postman[8] 断言

1.常见的断言类型 status code: code is 200 //检查返回的状态码是否为200 Response body: contain string //检查响应中包含指定字符串包含指定的值 response body:json value check/ /检查响应中其中json的值 Response body: is equal to string …

python openyxl 用法 教程

Python自动化办公:openpyxl教程(基础)-CSDN博客 https://zhuanlan.zhihu.com/p/342422919 https://openpyxl-chinese-docs.readthedocs.io/zh-cn/latest/tutorial.html 列标题,是这一列 对应的单元格的格式,默认是常规,设置之后…

深入解析 Wireshark 的 TLS 设置:应用场景与实操技巧

简述 在网络数据分析中,传输层安全(TLS)协议的流量解密和分析是一项重要的技能。Wireshark 提供了专门的设置选项,帮助用户处理 TLS 流量,例如解密会话、重组分片等。本文将详细解析上图所示的 Wireshark TLS 设置功能…

每天五分钟机器学习:凸集

本文重点 在SVM中,目标函数是一个凸函数,约束集合是一个凸集。因此,SVM问题可以转化为一个凸规划问题来求解。这使得SVM在实际应用中具有较高的计算效率和准确性。 凸集的定义 凸集是指一个集合中的任意两点之间的线段都完全包含在这个集合中。换句话说,给定集合C中的两…

stm32 智能语音电梯系统

做了个stm32智能语音控制的电梯模型,总结一下功能,源码用ST的HAL库写的,整体流程分明。 实物图 这个是整个板子的图片,逻辑其实并不复杂,只是功能比较多,在我看来都是一些冗余的功能,但也可能是…

AI 助力游戏开发中的常用算法实现

在当今的游戏开发领域,人工智能(AI)技术的应用已经成为推动行业发展的关键力量。AI不仅能够提升游戏的智能化水平,还能够增强玩家的沉浸感和游戏体验。随着技术的进步,AI在游戏设计、开发和测试中的应用越来越广泛&…

计算机的错误计算(一百九十九)

摘要 用大模型判断下面四个函数 有何关系?并计算它们在 x0.00024时的值,结果保留10位有效数字。两个大模型均认为它们是等价的。实际上,还有点瑕疵。关于计算函数值,大模型一只是纸上谈兵,没计算;大模型二…

HTML——57. type和name属性

<!DOCTYPE html> <html><head><meta charset"UTF-8"><title>type和name属性</title></head><body><!--1.input元素是最常用的表单控件--><!--2.input元素不仅可以在form标签内使用也可以在form标签外使用-…

基于SpringBoot和OAuth2,实现通过Github授权登录应用

基于SpringBoot和OAuth2&#xff0c;实现通过Github授权登录应用 文章目录 基于SpringBoot和OAuth2&#xff0c;实现通过Github授权登录应用0. 引言1. 创建Github应用2. 创建SpringBoot测试项目2.1 初始化项目2.2 设置配置文件信息2.3 创建Controller层2.4 创建Html页面 3. 启动…

LVGL 移植到 Arduino IDE(适用SP32 Arduino RP系列)

1.因为我们需要移植相关LVGL配置文件&#xff0c;否则IDE会报错&#xff0c;因此 先找到LVGL官方的GITHUB处&#xff0c;如下图所示&#xff1a; 2.值得注意的是&#xff0c;你需要知你的 Arduino IDE 用的是哪个版本的LVGL&#xff0c;要与我们在GITHUB处的版本号一致&#xf…

Ubuntu 24.04 LTS 解决网络连接问题

1. 问题描述 现象&#xff1a;ens33 网络接口无法获取 IPv4 地址&#xff0c;导致网络不可用。初步排查&#xff1a; 运行 ip a&#xff0c;发现 ens33 接口没有分配 IPv4 地址。运行 ping www.baidu.com&#xff0c;提示“网络不可达”。查看 NetworkManager 日志&#xff0c…

C语言----指针数组

目录 1. 定义&#xff1a; 2. 格式&#xff1a; 应用示例 1) 用于存放普通变量的地址 2) 用于存放二维数组的每一行第一个元素的地址&#xff08;列地址&#xff09; 3) 用于存放字符串 4) 命令行参数 补充&#xff1a;开辟堆区空间&#xff08;动态空间开辟&#xff0…