[C#.NET 拾遗补漏]06:单例模式最佳实践
阅读本文大概需要 3 分钟。
大家好,这是【C#.NET 拾遗补漏】专辑的第 06 篇文章。今天讲讲大家熟悉的单例模式。
单例模式大概是所有设计模式中最简单的一种,如果在面试时被问及熟悉哪些设计模式,你可能第一个答的就是单例模式。
单例模式的实现分为两种:饿汉式和懒汉式。前者是在静态构造函数执行时就立即实例化,后者是在程序执行过程中第一次需要时再实例化。两者有各自适用的场景,实现方式也都很简单,唯一在设计时要考虑的一个问题就是:实例化时需要保证线程安全。
饿汉式
饿汉式实现很简单,在静态构造函数中立即进行实例化:
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(); }}
}
这样会导致每次调用都会创建一个新实例。
懒汉式
懒汉式单例实现需要考虑线程安全问题,先来看一段经典的线程安全的单列模式实现代码:
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<T>
可以使对象的实例化延迟到第一次被调用的时候执行,通过访问它的 Value 属性来创建并获取实例,并且读取一个 Lazy<T>
实例的 Value 属性只会执行一次实例化代码,确保了线程安全。
祝,编码愉快!
[C#.NET 拾遗补漏]06:单例模式最佳实践相关推荐
- arcore之路-unity开发从入门到实践_Unity游戏开发——单例模式的最佳实践
0.前言 StarryFun:Unity游戏开发--关于单例模式的理解zhuanlan.zhihu.com 之前一篇文章讲了单例模式的简单理解,自知其中有很多不严谨的地方,由于本萌新也是在学习阶段, ...
- [翻译] DSL和模型驱动开发的最佳实践(3/4)
哪个是最佳选择 有两种风格的语言设计:一种主张大语言,用一个类支持许多不同的领域概念.另外一种主张小语言,使用一些小但是强大的原始的特征,大一些的特征由库的方式组装构建. 在 ...
- RESTful服务最佳实践
本文主要读者 引言 REST是什么 统一接口 基于资源 通过表征来操作资源 自描述的信息 超媒体即应用状态引擎(HATEOAS) 无状态 可缓存 C-S架构 分层系统 按需编码(可选) REST快速提 ...
- 调优 DB2 UDB v8.1 及其数据库的最佳实践-IBM developerWorks
http://www-128.ibm.com/developerworks/cn/db2/library/techarticles/dm-0404mcarthur/#sqltuning 级别: 初级 ...
- (流式、lambda、触发器)实时处理大比拼 - 物联网(IoT)\金融,时序处理最佳实践
标签 PostgreSQL , 物联网 , 传感器 , lambda , 调度 , 实时 , 流式更新 , UPSERT , insert on conflict do update 背景 越来越多的 ...
- 最佳实践 ADO.NET实用经验无保留曝光
ADO.NET作为微软最新的数据访问技术,已经在企业开发中得到了广泛的应用.对于一线的开发人员来说,掌握基本的概念和技术之后,提高应用水平和解决实际问题的最有效手段,莫过于相互交流彼此的最佳时间经验经 ...
- 钱大妈数据中台建设最佳实践
简介:钱大妈数据中台建设最佳实践 公司简介 钱大妈是在社区生鲜连锁中,以"不卖隔夜肉"作为品牌理念的的行业开拓者.在成立之初即从新鲜角度重新梳理传统生鲜行业的标准,对肉菜市场进行新 ...
- .NET Core开发实战(第15课:选项框架:服务组件集成配置的最佳实践)--学习笔记...
15 | 选项框架:服务组件集成配置的最佳实践 这一节讲解如何使用选项框架来处理服务和配置的关系 选项框架的特性: 1.支持单例模式读取配置 2.支持快照 3.支持配置变更通知 4.支持运行时动态修改 ...
- .NetCore 2.1以后的HttpClient最佳实践
ASP.NET Core 2.1中出现一个新的HttpClientFactory功能, 它有助于解决开发人员在使用HttpClient实例从其应用程序发出外部Web请求时可能遇到的一些常见问题. 01 ...
最新文章
- null未定义_PHP的isset()、is_null、empty()使用总结
- 如何解决ALV的负数符号前显的问题
- java db 程序_JAVA连接数据库的简单程序
- Linux 命令[3]:cd
- spring中bean的高级属性之list, set, map以及props元素(含举例)
- protocol buffer的高效编码方式
- 19.删除链表的倒数第N个节点
- CV Code | 计算机视觉开源周报20190904期
- java安装后启动程序在哪_java – 有没有办法在安装后使用一些“帮助应用程序”立即启动应用程序?...
- 【kafka】kafka 2.3 关于控制Broker端入站连接数的讨论
- 小程序中canvas绘制网络图片
- hdu 5288 OO’s Sequence(计数)
- 基于C#窗体的酒店管理系统
- 邮件服务器搭建 iredmail
- 小白学测试入门之测试分析方法
- linux ntp时间同步失败,linux下用ntpdate时间同步不成功。
- tf-encrypted介绍和代码示例
- 视频转文字怎么操作?这些方法值得收藏
- mac os 录屏快捷键_5款好用的Mac录屏软件推荐
- 自激多谐振荡电路实验总结,小白电路测试