单例模式以及在C#中的使用
下面做一些简要的说明。
1.
单例模式(Singleton Pattern),又称作单件模式,当然也有一种诙谐的称谓:单身模式。在经典的GoF所著的《Design Patterns》一书中,对单例模式有着详尽的介绍,这本书网上有全文版本。
2.
单例模式的意图是保证一个类仅有一个实例,并且要提供一个全局访问点来访问这个实例。通常这个全局访问点是一个静态方法或者C#中的一个属性。
3.
在C#中,典型的单例模式实现方法可以如下:
public class Manager { private static Manager Mgr; //constructor must be private private Manager() { } //public and static, or you can change it to Property instead of Method. public static Manager GetInstance() { if (Mgr == null ) { Mgr = new Manager(); } return Mgr; } } |
当外部需要Manager
的实例时,可以调用GetInstance()
这个静态方法。由于Manager
类的构造器是私有的,这也就避免了其他方式实例化这个Manager
类。GetInstance()
方法内部的实现,保证了全局中只有一个Manager
实例。
4.
问题肯定不会这么简单就被解决,比如在多线程环境中,上述代码就会有很大的隐患。
- 有一种情况很常见:两个线程同时调用
GetInstance()
方法; - 当某一个线程由于
Mgr
为null
而进入条件判断代码块的时候,而恰恰还没有执行实例化一个Manager
对象,这时候另一个线程由于Mgr
为null
,所以也会进入这个条件语句中
上面两种情况,很显然都会创建Manager
实例,这也就违背了单例模式的意图了。
利用C#的特性,我们可以把一个线程先锁住(lock
),等到这个线程完成后,再让下一个线程访问GetInstance()
方法:
private static readonly object syncObject = new object (); //public and static, or you can change it to Property instead of Method. public static Manager GetInstance() { //double check if (Mgr == null ) { lock (syncObject) { if (Mgr == null ) { Mgr = new Manager(); } } } return Mgr; } |
代码中用到了双重检查锁定(double check locking)的技术,是为了提高性能考虑,因为C#中lock
语句是很耗性能的。第一道检查,是基于如果Mgr
不为null
的时候就不需要lock
了,提高性能。第二道检查,是基于两个线程同时通过第一道检查后,第一个线程解锁后,由于Mgr
此时已经不为null
,所以第二个线程就不用实例化Manager
了。
5.
单例模式有两种实现方式,主要基于构建的方式不同:
- 延迟初始化(Lazy Initialization),也叫“懒汉模式”:单例实例在第一次使用时被构建;
- 热初始化(Eager Initialization),也叫“恶汉模式”:单例实例在类加载时创建
前面创建单例模式的方式都属于延迟初始化。.NET 4.0以后提供了一个Lazy<T>
泛型类,可以被应用于这个场景,省却代码的编写量。
public class Manager { private static Lazy<Manager> mgr = new Lazy<Manager>(); //version of Thread Safe //private static Lazy<Manager> mgr = new Lazy<Manager>(true); private Manager() { } public static Manager GetInstance() { return mgr.Value; } } |
Lazy<T>
的构造器重载版本可以帮我们解决多线程的问题。
C#使用静态初始化来完成单例模式中的热初始化。需要注意的是,不需要考虑多线程的问题,因为CLR会自动解决多线程同步的问题。如果程序经常要用到这个实例,运用热初始化可以显著提高性能。
public class Manager { private static readonly Manager mgr = new Manager(); private Manager() { } public static Manager GetInstance() { return mgr; } } |
6.
StackOverflow中对于单例模式都是持否定态度的,比如这个:
In theory: when you need to restrict the instantiation of an object to one instance. In practice: never.
主要基于下面几个原因:
- 违反了单一职责原则
- 耦合度过大
- 单元测试基本无法进行
- 开发混淆,造成混乱。比如作为API提供的时候。
总之,
There‘s at most a can use but there no need.
文章来源:http://laobian.me/2013/02/singleton-pattern-with-csharp.html
单例模式以及在C#中的使用的更多相关文章
- 单例模式@Singleton在测试中的运用
前言 单例模式是一种比较常用的设计模式,目的是:保证一个类仅有一个实例,并提供一个访问它的全局访问点. 测试种可能用到的场景 : 在很多时候,有些对象我们希望在整个程序只有一个实例,如线程池.数据库连 ...
- 单例模式读取properties配置文件中的信息
public class ConfigManager { private static ConfigManager config = null; //创建Properties文件 读取配 ...
- Objective-C中的单例模式
单例模式算是设计模式中比较简单的一种吧,设计模式不是只针对某种编程语言,在C++, Java, PHP等其他OOP语言也有设计模式,笔者初接触设计模式是通过<漫谈设计模式>了解 ...
- Java中的GOF23(23中设计模式)--------- 单例模式(Singleton)
Java中的GOF23(23中设计模式)--------- 单例模式(Singleton) 在Java这这门语言里面,它的优点在于它本身的可移植性上面,而要做到可移植的话,本身就需要一个中介作为翻译工 ...
- Android中的单例模式
定义: 单例模式:确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 使用场景: 确保某一个类有且只有一个对象的场景,避免产生多个对象消耗过多的资源,或者某种类型的对象只应该有且只有一 ...
- java中的单例模式与doublecheck
转自: http://devbean.blog.51cto.com/448512/203501 在GoF的23种设计模式中,单例模式是比较简单的一种.然而,有时候越是简单的东西越容易出现问题.下面就单 ...
- java中的单例模式与静态类
单例模式与静态类(一个类,所有方法为静态方法)是另一个非常有趣的问题,在<Java中有关单例模式的面试问题>博文中露掉了,由于单例模式和静态类都具有良好的访问性,它们之间有许多相似之处,例 ...
- 设计模式 - Java中单例模式的6种写法及优缺点对比
目录 1 为什么要用单例模式 1.1 什么是单例模式 1.2 单例模式的思路和优势 2 写法① - 饥饿模式 2.1 代码示例 2.2 优缺点比较 3 写法② - 懒惰模式 3.1 代码示例 3.2 ...
- 设计模式(五): 简单而又不失其重要性的单例模式(Singleton Pattern)
上篇博客我们系统的介绍了三种工厂模式,今天我们就来介绍一下单例模式.单例模式虽然简单,但是还是比较重要的,是常用设计模式之一.在之前的博客<Objective-C中的单例模式>中介绍了Ob ...
随机推荐
- soj114 中位数
题意:给你一个长度为2n-1的数组A,设Bi是A的1~2i-1的中位数.问打乱A,有多少种不同的B序列? 标程: #include<bits/stdc++.h> using namespa ...
- [JZOJ3339]【NOI2013模拟】wyl8899和法法塔的游戏
题目 题目大意 给你一个数列,每次给出\(r,a,b\),你要找到\(l\in [a,b]\)使得\([l,r-1]\)的异或和最小, 并且要修改\(r\)位置的数. 思考历程 当我看到这题的时候,已 ...
- idea从github中pull或者push成功之后tomcat启动不了,报Error....
解决方案:删除deployment里面的war包,确定. 再在deployment里面重新添加一个war包,确定,即可.
- 如何在 JavaScript 中使用 C 程序
JavaScript 是个灵活的脚本语言,能方便的处理业务逻辑.当需要传输通信时,我们大多选择 JSON 或 XML 格式. 但在数据长度非常苛刻的情况下,文本协议的效率就非常低了,这时不得不使用二进 ...
- PAT甲级——A1120 Friend Numbers【20】
Two integers are called "friend numbers" if they share the same sum of their digits, and t ...
- k8s 是什么,有什么功能
k8s是一个docker集群的管理工具 k8s是容器的编排工具 1. k8s的核心功能 自愈: 重新启动失败的容器,在节点不可用时,替换和重新调度节点上的容器,对用户定义的健康检查不响应党的容器会被中 ...
- SGLTE/SVLTE、CSFB、SRVCC概念
SGLTE:Simultaneous GSM and LTE,手机可以同时驻留在GSM和LTE网络中,打电话通过GSM网络进行,数据业务通过LTE网络进行. SVLTE:Simultaneous V ...
- Python - 作为浅拷贝的list对象乘法
运行下面这段代码 # !/usr/bin/env python3 # -*- coding=utf-8 -*- temp_a = [[0]*2]*3 temp_b = [[0]*2 for i in ...
- java-day07
API 应用程序编程接口 Scanner类 将键盘输入的数据到程序中 1.导包 2.创建 Scanner 对象名 = new Scanner(); 3.使用 int num = 对象名.nextInt ...
- [课后作业] 第002讲:用Python设计第一个游戏 | 课后测试题
试题: 0. 什么是BIF? 1. 用课堂上小甲鱼教的方法数一数 Python3 提供了多少个 BIF? 2. 在 Python 看来:'FishC' 和 'fishc' 一样吗? 3. 在小甲鱼看来 ...