单例模式(Singleton):保证一个类仅有一个实例,并提供一个访问它的全局访问点。

一、单例模式

  通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象。一个最好的办法就是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。

  单例模式结构图:

  

  Singleton类,定义一个GetInstance操作,允许客户访问它的唯一实例。GetInstance是一个静态方法,主要负责创建自己的唯一实例。

  单例模式示例:

namespace ConsoleApplication1
{
public class Singleton
{
private static Singleton instance; private Singleton() //将构造方法设为private,防止外界利用new创建实例。
{ } public static Singleton GetInstance() //此方法用于获得本类实例的唯一全局访问点
{
if (instance == null) //若实例不存在则new一个新实例,否则返回已有实例
{
instance = new Singleton();
}
return instance;
}
}
class Program
{
static void Main(string[] args)
{
Singleton singleton1 = Singleton.GetInstance();
Singleton singleton2 = Singleton.GetInstance(); //调用静态方法获得实例
if (singleton1 == singleton2)
{
Console.WriteLine("两个对象是相同的实例!"); //输出 两个对象是相同的示例
}
Console.ReadKey();
}
}
}

  单例模式与实用类(静态方法)的不同点:

    1、实用类不保存状态,仅提供一些静态方法或静态属性让你用,而单例是由状态的。

    2、实用类不能用于继承多态,而单例虽然实例唯一,却是可以有子类来继承。

    3、实用类只不过是一些方法属性的集合,而单例却是有着唯一的对象实例。

二、多线程时的单例

  在上面的例子中,如果是在多线程的程序中,多个线程同时(注意是同时)访问Singleton类,调用GetInstance()方法,会有可能造成创建多个实例的。因此可以通过加lock来处理,lock是确保当一个线程位于代码的临界区,另一个线程不进入临界区。如果其他线程视图进入锁定的代码,则它将一直等待(即被阻止),直到对象被释放。

  双重锁定代码示例:

namespace ConsoleApplication1
{
public class Singleton
{
private static Singleton instance;
private static readonly object syncRoot = new object(); private Singleton() //将构造方法设为private,防止外界利用new创建实例。
{ } public static Singleton GetInstance() //此方法用于获得本类实例的唯一全局访问点
{
if (instance == null) //如果实例为空,在外面做一层判断的作用是,当实例不为null时,不用lock,减少资源消耗。
{
lock (syncRoot) //锁定
{
if (instance == null) //当两个线程同时调用GetInstance()时,他们都可以通过第一重instance=null的判断,然后由于lock机制,
//这两个线程只有一个进入,另一个等候,如果没有第二重的instance=null的判断,则第一个线程创建了实例,第二个线程由于过了第一重instance=null,
//因此在没有第二重instance == null判断的情况下还是会创建实例。
{
instance = new Singleton();
}
}
}
return instance;
}
}
class Program
{
static void Main(string[] args)
{
Singleton singleton1 = Singleton.GetInstance();
Singleton singleton2 = Singleton.GetInstance(); //调用静态方法获得实例
if (singleton1 == singleton2)
{
Console.WriteLine("两个对象是相同的实例!"); //输出 两个对象是相同的示例
}
Console.ReadKey();
}
}
}

三、静态初始化

  其实在实际应用中,C#与公共语言运行库也提供了一种‘静态初始化’方法,这种方法不需要开发人员显式地编写线程安全代码,既可解决多线程环境下它是不安全的问题。相对于上面的来说,两种方法都可以达到相同的单例目的,只是实现更加简单而已,来看下实现代码:

namespace ConsoleApplication1
{
public sealed class Singleton //注意必须设置为密封类
{
private static readonly Singleton instance = new Singleton(); //注意声明为只读
private Singleton()
{ }
public static Singleton GetInstance()
{
return instance;
}
}
class Program
{
static void Main(string[] args)
{
Singleton singleton1 = Singleton.GetInstance();
Singleton singleton2 = Singleton.GetInstance(); //调用静态方法获得实例
if (singleton1 == singleton2)
{
Console.WriteLine("两个对象是相同的实例!"); //输出 两个对象是相同的示例
}
Console.ReadKey();
}
}
}

  这种实现与前面的示例类似,也是解决了单例模式试图解决的两个基本问题:全局访问和实例化控制,公共静态属性为访问实例提供了一个全局访问点。不同之处在于它依赖公共语言运行库来初始化变量。由于构造方法是私有的,因此不能在类本身以外实例化Singleton类;因此,变量引用的是可以再系统中存在的唯一实例。不过要注意,instance变量标记为readonly,这意味着只能在静态初始化期间或在类的构造函数中分配变量。由于这种静态初始化的方式是在自己被加载时就将自己实例化,所以被形象地称之为饿汉单例模式。而上面的第二种是在第一次被引用的时候,才会将自己实例化,所以就被称之为懒汉单例模式

  由于饿汉式,即静态初始化的方式,它是类一加载就实例化对象,所以要提前占用系统资源。然而懒汉式,又会面临着多线程访问的安全性问题,需要做双重锁定这样的处理才可以保证安全。所以到底使用哪一种方式取决于实际的需求。从C#语言的角度来讲,饿汉式的单例模式已经足够满足我们的需求。

单例模式 - OK的更多相关文章

  1. C++实现线程安全的单例模式

    在某些应用环境下面,一个类只允许有一个实例,这就是著名的单例模式.单例模式分为懒汉模式,跟饿汉模式两种. 首先给出饿汉模式的实现 template <class T> class sing ...

  2. 23种设计模式--单例模式-Singleton

    一.单例模式的介绍 单例模式简单说就是掌握系统的至高点,在程序中只实例化一次,这样就是单例模式,在系统比如说你是该系统的登录的第多少人,还有数据库的连接池等地方会使用,单例模式是最简单,最常用的模式之 ...

  3. angular2系列教程(十)两种启动方法、两个路由服务、引用类型和单例模式的妙用

    今天我们要讲的是ng2的路由系统. 例子

  4. java设计模式之--单例模式

    前言:最近看完<java多线程编程核心技术>一书后,对第六章的单例模式和多线程这章颇有兴趣,我知道我看完书还是记不住多少的,写篇博客记录自己所学的只是还是很有必要的,学习贵在坚持. 单例模 ...

  5. 设计模式C#合集--单例模式

    单例模式 代码: 第一种: private static Singleton singleton = null; private Singleton() { } public static Singl ...

  6. 设计模式之单例模式(Singleton)

    设计模式之单例模式(Singleton) 设计模式是前辈的一些经验总结之后的精髓,学习设计模式可以针对不同的问题给出更加优雅的解答 单例模式可分为俩种:懒汉模式和饿汉模式.俩种模式分别有不同的优势和缺 ...

  7. GOF23设计模式之单例模式

    ·核心作用: -保证一个类只有一个实例,并且提供一个访问该实例的全局访问点. ·常见应用场景: -Windows的Task Manager(任务管理器)就是很典型的单例模式 -Windows的Recy ...

  8. GJM : C#设计模式(1)——单例模式

    感谢您的阅读.喜欢的.有用的就请大哥大嫂们高抬贵手"推荐一下"吧!你的精神支持是博主强大的写作动力以及转载收藏动力.欢迎转载! 版权声明:本文原创发表于 [请点击连接前往] ,未经 ...

  9. PHP设计模式(四)单例模式(Singleton For PHP)

    今天讲单例设计模式,这种设计模式和工厂模式一样,用的非常非常多,同时单例模式比较容易的一种设计模式. 一.什么是单例设计模式 单例模式,也叫单子模式,是一种常用的软件设计模式.在应用这个模式时,单例对 ...

  10. java设计模式之单例模式(几种写法及比较)

    概念: Java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建 ...

随机推荐

  1. 【翻译】五步快速使用LINQPad尝鲜StreamInsight

    StreamInsight  学习地址:http://www.cnblogs.com/StreamInsight/archive/2011/10/26/StreamInsight-Query-Seri ...

  2. CF-599B - Spongebob and Joke

    B. Spongebob and Joke time limit per test 2 seconds memory limit per test 256 megabytes input standa ...

  3. HDU-1078

    Problem Description FatMouse has stored some cheese in a city. The city can be considered as a squar ...

  4. 40 个重要的 HTML5 面试问题及答案

    本文将列出40个重要的HTML 5面试问题及答案,祝各位求职顺利. 目录 介绍 Canvas和SVG图形之间的区别是什么? 如何使用Canvas和SVG绘制矩形? 什么是CSS选择器? 如何使用ID值 ...

  5. 远程连接mysql

    win系统下,连接别人的mysql或者让别人链接自己的mysql: 打开命令行cmd 进入mysql: mysql -u root -p mysql>use mysql;  mysql>s ...

  6. solr索引

    solr索引 当我们真正进入到Lucene源代码之中的时候,我们会发现: • Lucene的索引过程,就是按照全文检索的基本过程,将倒排表写成此文件格式的过程. • Lucene的搜索过程,就是按照此 ...

  7. SpringMVC拦截器(实现登录验证拦截器)

    本例实现登陆时的验证拦截,采用SpringMVC拦截器来实现 当用户点击到网站主页时要进行拦截,用户登录了才能进入网站主页,否则进入登陆页面 核心代码 首先是index.jsp,显示链接 <%@ ...

  8. 【C#正则基础】正则表达式

    1. 代表任意多个字符:(.*?)2. 代表网页里的<body>*</body>任意的标签内容,替换以后网页源码就只剩纯文本:<[^>]*>3. 代表网页中的 ...

  9. OpenCV中OpenCL模块函数

    It currently develop and test on GPU devices only. This includes both discrete GPUs(NVidia,AMD), as ...

  10. 金典 SQL笔记(4)

    由于在本地笔记上写的.CSDN markdown编辑器仅仅支持.md格式导入, 图片没办法直接导进去.写的多了懒的一张一张图片切图上传; 直接整个文章切成图片上传上去了. watermark/2/te ...