C#--设计模式之单例模式
单例模式大概是所有设计模式中最简单的一种,如果在面试时被问及熟悉哪些设计模式,你可能第一个答的就是单例模式。
单例模式的实现分为两种:
- 饿汉式:在静态构造函数执行时就立即实例化。
- 懒汉式:在程序执行过程中第一次需要时再实例化。
两者有各自适用的场景,实现方式也都很简单,唯一在设计时要考虑的一个问题就是:实例化时需要保证线程安全。
1. 饿汉式
饿汉式实现很简单,在静态构造函数中立即进行实例化:
public class Singleton
{private static readonly Singleton _instance;static Singleton(){_instance = new Singleton();}public static Singleton Instance{get{return _instance;}}
}
注意,为了确保单例性,需要使用 readonly 关键字声明实例不能被修改。
以上写法可简写为:
public class Singleton
{private static readonly Singleton _instance = new Singleton();public static Singleton Instance{get{return _instance;}}
}
这里的 new Singleton() 等同于在静态构造函数中实例化。在 C# 7 中还可以进一步简写如下:
public class Singleton
{public static Singleton Instance { get; } = new Singleton();
}
一句代码就搞定了,此写法,实例化也是在默认的静态构造函数中进行的。如果是饿汉式需求,这种实现是最简单的。有人会问这会不会有线程安全问题,如果多个线程同时调用 Singleton.Instance 会不会实例化了多个实例。不会,因为 CLR 确保了所有静态构造函数都是线程安全的。
注意,不能这么写:
public class Singleton
{public static Singleton Instance => new Singleton();
}
// 等同于:
public class Singleton
{public static Singleton Instance{get { return new Singleton(); }}
}
这样会导致每次调用都会创建一个新实例。
2. 懒汉式
懒汉式单例实现需要考虑线程安全问题,先来看一段经典的线程安全的单列模式实现代码:
public sealed class Singleton
{private static volatile Singleton _instance;private static readonly object _lockObject = new Object();public static Singleton Instance{get{if (_instance == null){lock (_lockObject){if (_instance == null){_instance = new Singleton();}}}return _instance;}}
}
网上搜索 C# 单例模式,大部分都是这种使用 lock 来确保线程安全的写法,这是经典标准的单例模式的写法,没问题,很放心。在 lock 里外都做一次 instance 空判断,双保险,足以保证线程安全和单例性。但这种写法似乎太麻烦了,而且容易写错。早在 C# 3.5 的时候,就有了更好的写法,使用 Lazy<T>。
示例代码:
public class LazySingleton
{private static readonly Lazy<LazySingleton> _instance =new Lazy<LazySingleton>(() => new LazySingleton());public static LazySingleton Instance{get { return _instance.Value; }}
}
调用示例:
public class Program
{public static void Main(){var instance = LazySingleton.Instance;}
}
使用 Lazy 可以使对象的实例化延迟到第一次被调用的时候执行,通过访问它的 Value 属性来创建并获取实例,并且读取一个 Lazy 实例的 Value 属性只会执行一次实例化代码,确保了线程安全。
3. 应用示例
static void Main(string[] args)
{LazySingleton p1 = LazySingleton.Instance();p1.GetName();LazySingleton p2 = LazySingleton.Instance();p2.GetName();if (p1 == p2){Console.WriteLine("他们是同一个人!");}else{Console.WriteLine("他们不是同一人!");}Console.ReadLine();
}public class LazySingleton
{private static volatile LazySingleton instance = null;//保证instance在所有线程中同步//private防止类在外部被实例化private LazySingleton(){ }public static LazySingleton Instance(){if (instance == null){instance = new LazySingleton();}return instance;}public void GetName(){Console.WriteLine("詹姆斯");}
}
程序运行结果:

相关文章:
C#--设计模式之单例模式
单例模式大概是所有设计模式中最简单的一种,如果在面试时被问及熟悉哪些设计模式,你可能第一个答的就是单例模式。 单例模式的实现分为两种: 饿汉式:在静态构造函数执行时就立即实例化。懒汉式:在程序执行过程中第一…...
RWEQ风蚀方程模型与ArcGIS数据处理Python代码库添加结合理论研究和科研实践
RWEQ模型是应用比较普遍的能适应大区域定量估算风蚀量的模型。该模型是基于大量野外实验的一种经验模型,在实际测定风力导致的土壤侵蚀量以及当地的气象、地表植被、土壤湿度、地表的结皮和地表的可蚀性等因子的基础上得出的一个经验方程。 1、掌握土壤风蚀模型的原…...
基于STM32微控制器的物联网(IoT)节点设计与实现
基于STM32微控制器的物联网(IoT)节点的设计和实现。我们讨论物联网节点的基本概念和功能,并详细介绍了STM32微控制器的特点和优势。然后,我们将探讨如何使用STM32开发环境和相关的硬件模块来设计和实现一个完整的物联网节点。最后,我们将提供一个示例代码,展示如何在STM3…...
篇二十一:中介者模式:解耦对象之间的交互
篇二十一:"中介者模式:解耦对象之间的交互" 开始本篇文章之前先推荐一个好用的学习工具,AIRIght,借助于AI助手工具,学习事半功倍。欢迎访问:http://airight.fun/。 另外有2本不错的关于设计模式…...
tomcat的多实例,动静分离(web服务基础结束)
多实例 多实例就是在一台服务器上有多个tomcat的服务(核心是改端口) 实验:多实例 安装步骤 1.安装好 jdk 2.安装 tomcat cd /opt tar zxvf apache-tomcat-9.0.16.tar.gz mkdir /usr/local/tomcat mv apache-tomcat-9.0.16 /usr/local/tomca…...
LeetCode150道面试经典题--判断子序列(简单)
1.题目 给定字符串 s 和 t ,判断 s 是否为 t 的子序列。 字符串的一个子序列是原始字符串删除一些(也可以不删除)字符而不改变剩余字符相对位置形成的新字符串。(例如,"ace"是"abcde"的一个子序…...
kubeadml 安装 k8s
目录 一:kubeadml 安装 k8s 1、网络环境 2、 环境准备 3、 所有节点安装docker 4、所有节点安装kubeadm,kubelet和kubectl 5、部署K8S集群 6、测试 二: 部署 Dashboard 一:kubeadml 安装 k8s 1、网络环境 master&am…...
考研C语言进阶题库——更新16-20题
目录 16计算t11/2...1/n-11/n 17计算1997! 18计算t1-122-133-...-1nn 19相传国际象棋是古印度舍罕王的宰相达依尔发明的.舍罕王十分喜爱象棋,决定让宰相自己选择何种赏赐. 这位聪明的宰相指着8*8共64格的象棋说:陛下,请您赏给我一些麦子吧. 就在棋盘的第1格放1粒…...
【变形金刚01】attention和transformer所有信息
图1.来源:Arseny Togulev在Unsplash上的照片 一、说明 这是一篇 长文 ,几乎讨论了人们需要了解的有关注意力机制的所有信息,包括自我注意、查询、键、值、多头注意力、屏蔽多头注意力和转换器,包括有关 BERT 和 GPT 的一些细节。因…...
面试热题(路径总和II)
给你二叉树的根节点 root 和一个整数目标和 targetSum ,找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径。 叶子节点 是指没有子节点的节点。 在这里给大家提供两种方法进行思考,第一种方法是递归,第二种方式使用回溯的方式进行爆…...
测试 tensorflow 1.x 的一个demo 01
tensorflow 1.0的示例代码 demo_01.py import tensorflow as tf import os os.environ[TF_CPP_MIN_LOG_LEVEL]2def tf114_demo():a 3b 4c a bprint("a b in py ",c)a_t tf.constant(3)b_t tf.constant(4)c_t a_t b_tprint("TensorFlow add a_t b_t &…...
达蒙DM数据库使用经验
DM表/字段注释 注:dm数据库无法在建表的同时为字段名添加注释 //为表添加注释 comment on table 库名.表名 is 表注释; //为表字段添加注释 comment on column 库名.表名.列名 is 列注释;DM查询错误:无效的表或视图 1,确认表一定存在 2&am…...
Redis—集群
目录标题 主从复制第一次同步命令传播分担主服务器压力增量复制总结面试题什么是Redis主从复制Redis主从复制的原理Redis主从复制的优点Redis主从复制的缺点Redis主从复制的配置步骤Redis主从复制的同步策略主从节点是长还是短连接判断某个节点是否正常工作主从复制架构中&…...
【C语言】数据在内存中的存储详解
文章目录 一、什么是数据类型二、类型的基本归类三、 整型在内存中的存储1.原码、反码、补码2.大小端(1)什么是大小端(2)为什么会有大小端 四、浮点型在内存中的存储1. 浮点数存储规则 五、练习1.2.3.4.5.6.7. 一、什么是数据类型 我们可以把数据类型想象为一个矩形盒子&#x…...
PIC单片机配置字的设置
PIC单片机配置字的设置 PIC系列单片机,其芯片内部大都设置有一个特殊的程序存储单元,地址根据不同的单片机而定,此存储单元用来由单片机用户自由配置或定义单片机内部的一些功能电路单元的性能选项,所以被称之为系统配置字。目前PIC单片机系统配置字的方法有两种,一种是利…...
JavaWeb-Servlet服务连接器(一)
目录 1.Servlet生命周期 2.Servlet的配置 3.Servlet的常用方法 4.Servlet体系结构 5.HTTP请求报文 6.HTTP响应报文 1.Servlet生命周期 Servlet(Server Applet)是Java Servlet的简称。其主要的功能是交互式地浏览和修改数据,生成一些动态…...
新华三超融合态势感知标准版
产品概述: H3C SecCenter CSAP-XS 超融合态势感知一体机产品集合了态势感知和安全流量分析探针设备能无需复杂配置;态势感知平台具备强大的安全分析和可视化呈现功能;同时具备远程专家会诊功能,通过云端协同实现外部安全服务资源的…...
AutoSAR系列讲解(深入篇)13.2-Mcal Port配置
目录 一、配置界面 二、通用配置 1、ConfigVariant 2、PortSafety 3、PortGeneral 三、Port配置集合...
Java旋转数组中的最小数字(图文详解版)
目录 1.题目描述 2.题解 分析 具体实现 方法一(遍历): 方法二(排序): 方法三(二分查找): 1.题目描述 有一个长度为 n 的非降序数组,比如[1,2,3,4,5]&a…...
Android 13 Hotseat定制化修改——005 hotseat图标禁止形成文件夹
目录 一.背景 二.方案 一.背景 由于需求是需要自定义修改Hotseat,所以此篇文章是记录如何自定义修改hotseat的,应该可以覆盖大部分场景,修改点有修改hotseat布局方向,hotseat图标数量,hotseat图标大小,hotseat布局位置,hotseat图标禁止形成文件夹,hotseat图标禁止移动…...
PS 抠完图怎么加外描边?超简单 3 种方法,零基础秒学会
做设计、电商配图、海报制作时,抠图只是基础步骤。给抠好的人物、产品、素材添加描边,既能强化主体轮廓、区分画面层次,还能提升整体视觉质感。但很多 PS 新手抠完图后,不知道怎么快速加描边,容易出现边缘锯齿、描边遮…...
从CMU15-445 Project#1出发:手把手教你用C++实现LRU-K缓存替换策略(附完整源码)
从零实现LRU-K缓存替换策略:CMU15-445 Project#1深度解析与C实战 在数据库系统与操作系统领域,缓存替换策略直接影响着系统性能。当CMU15-445课程Project#1要求实现LRU-K算法时,许多学习者发现原始论文晦涩难懂,而网上又缺乏完整…...
学术论文PDF怎么转结构化数据
做过文献调研的人都深有体会:面对成百上千篇PDF格式的学术论文,想要系统性地提取其中的数据、公式、表格,简直是一场噩梦。传统OCR工具不是把公式识别成乱码,就是把双栏排版的段落顺序彻底打乱。合合信息推出的TextIn文档解析&…...
poi-tl填坑实录:升级到1.10.x后,表格循环和复选框渲染策略变了怎么办?
poi-tl 1.10.x升级指南:表格循环与复选框渲染的深度适配方案 最近在重构一个企业级文档生成系统时,我遇到了一个典型的技术债问题——项目使用的poi-tl库长期停留在1.9.1版本,而新版本1.10.x对表格循环和复选框渲染机制做了重大调整。这导致原…...
如何永久保存微信聊天记录:WeChatMsg完整指南与数据掌控
如何永久保存微信聊天记录:WeChatMsg完整指南与数据掌控 【免费下载链接】WeChatMsg 提取微信聊天记录,将其导出成HTML、Word、CSV文档永久保存,对聊天记录进行分析生成年度聊天报告 项目地址: https://gitcode.com/GitHub_Trending/we/WeC…...
从Kaggle竞赛到业务落地:XGBoost分类实战中的5个关键参数陷阱与解决方案
从Kaggle竞赛到业务落地:XGBoost分类实战中的5个关键参数陷阱与解决方案 当你在Kaggle排行榜上看到XGBoost模型大杀四方,信心满满地将它部署到业务系统中时,是否遇到过这样的困惑:为什么同样的参数设置,在实际业务中的…...
Pearcleaner:重新定义macOS应用卸载的智能系统
Pearcleaner:重新定义macOS应用卸载的智能系统 【免费下载链接】Pearcleaner A free, source-available and fair-code licensed mac app cleaner 项目地址: https://gitcode.com/gh_mirrors/pe/Pearcleaner 你是否曾经思考过,当我们"删除&q…...
ESP8266玩转网络引导:搭建一个‘钓鱼Wi-Fi’式演示服务器(用于产品原型展示)
ESP8266打造无感化产品演示系统:从技术实现到商业场景落地 想象一下这样的场景:在熙熙攘攘的展会上,潜在客户只需用手机连接一个名为"Demo_Product"的Wi-Fi热点,打开浏览器输入"demo.product"——无需记忆IP地…...
3分钟掌握SRWE:免费窗口分辨率自定义终极指南
3分钟掌握SRWE:免费窗口分辨率自定义终极指南 【免费下载链接】SRWE Simple Runtime Window Editor 项目地址: https://gitcode.com/gh_mirrors/sr/SRWE Simple Runtime Window Editor(SRWE)是一款轻量级开源窗口分辨率工具࿰…...
Sunshine终极指南:三步搭建你的专属游戏串流服务器
Sunshine终极指南:三步搭建你的专属游戏串流服务器 【免费下载链接】Sunshine Self-hosted game stream host for Moonlight. 项目地址: https://gitcode.com/GitHub_Trending/su/Sunshine Sunshine是一款开源的自托管游戏串流服务器,专为Moonlig…...
