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

一、动机(Motivation)

在软件系统中,经常有这样一些特殊的类,必须保证它们在系统中只存在一个实例,才能确保它们的逻辑正确性、以及良好的效率。

如何绕过常规的构造器,提供一种机制来保证一个类只有一个实例?

这应该是类设计者的责任,而不是使用者的责任。

二、意图(Intent)

保证一个类仅有一个实例,并提供一个该实例的全局访问点

三、结构(Structure)

保证一个类仅有一个实例,并提供一个访问它的全局访问点。

四、单例模式的代码实现

1、单线程Singleton模式的实现

public sealed class Singleton
{
    private static Singleton uniqueInstance;// 定义一个静态变量来保存类的实例

    private Singleton() // 定义私有构造函数,使外界不能创建该类实例
    {
    }

    /// 

    /// 定义公有方法提供一个全局访问点,同时你也可以定义公有属性来提供全局访问点
    ///
    ///
    public static Singleton GetInstance()
    {
        if (uniqueInstance == null)// 如果类的实例不存在则创建,否则直接返回
        {
            uniqueInstance = new Singleton();
        }
        return uniqueInstance;
    }
}

私有的实例构造器是为了屏蔽默认产生的构造器,让类的使用者无法调用构造器。

单线程Singleton模式的几个要点

Singleton模式中的实例构造器可以设置为protected以允许子类派生。

Singleton模式一般不要支持ICloneable接口,因为这可能会导致多个对象实例,与Singleton模式的初衷违背。

Singleton模式一般不要支持序列化,因为这也有可能导致多个对象实例,同样与Singleton模式的初衷违背。

Singleton模式只考虑到了对象创建的管理,没有考虑对象销毁的管理。就支持垃圾回收平台和对象的开销来讲,我们一般没有必要对其销毁进行特殊的管理。

不能应对多线程环境:在多线程环境下,使用Singleton模式仍然有可能得到Singleton类的多个实例对象。

2、双多线程重锁定(Double Check)Singleton模式的实现

public sealed class Singleton
{
    private static volatile Singleton uniqueInstance;// volatile关键字知识此成员变量能被多个线程访问。

    private static readonly object locker = new object();// 定义一个标识确保线程同步

    private Singleton()// 定义私有构造函数,使外界不能创建该类实例
    {
    }

    public static Singleton GetInstance()
    {
        if (uniqueInstance == null)//先判断不存在再枷锁
        {
            lock (locker)
            {
                if (uniqueInstance == null)// 如果类的实例不存在则创建,否则直接返回
                {
                    uniqueInstance = new Singleton();
                }
            }
        }
        return uniqueInstance;
    }
}

volatile修饰符:编译器在编译代码的时候会对代码的顺序进行微调,用volatile修饰保证了严格意义的顺序。一个定义为volatile的变量是说这变量可能会被意想不到地改变,这样,编译器就不会去假设这个变量的值了。精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。

3、使用C#语言的“静态初始化”特性来实现单例的Singleton模式。(.Net中实现Singleton的首选方法)

内联初始化(生成的同时进行初始化):

public sealed class Singleton
{
    private static readonly Singleton instance = new Singleton();

    public static Singleton GetInstance()
    {
        return Singleton.instance;
    }

    private Singleton()//私有构造函数,防止外界调用
    {
       //...
     }
}

它等同于:

public sealed class Singleton
{
    public static readonly Singleton instance;

    //静态构造函数,初始化静态变量。CLR只执行一次
    static Singleton()
    {
        instance = new Singleton();
    }

    public static Singleton GetInstance()
    {
        return Singleton.instance;
    }

    private Singleton()//私有构造函数,防止外界调用
    {
       //...
    }
}

另一种优雅写法是要用到.net 4.0里Lazy

public sealed class Singleton
{
    private static readonly Lazy lazy =
        new Lazy(() => new Singleton());

    public static Singleton Instance => lazy.Value;

    private Singleton()
    {
    }
}

只要想访问静态字段,必定已经在之前执行了静态构造器。这样也能够精确地保证使用的时候一定能拿到实例,如果不使用也不会实例化对象,也就是延时加载的功能。他同样能够支持多线程环境,因为只可能有一个线程执行静态构造器,不可能有多个线程去执行静态构造器,感觉就是程序已经自动为我们加锁了。它的一点弊端就是它不支持参数化的实例化方法。

在.NET里静态构造器只能声明一个,而且必须是无参数的,私有的。因此这种方式只适用于无参数的构造函数。

五、.NET框架中的Singleton应用

t1==t2 这说明,GetType方法获得的Type实例都是单例。

HttpContext.Current也是如此,他们是通过Singleton的扩展方式实现的,他们的单例也并不是覆盖所有领域,只是针对某些局部领域中,是单例的,不同的领域中还是会有不同的实例。

到此这篇关于.Net设计模式之单例模式(Singleton)的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • .Net设计模式之工厂方法模式(Factory Method)

    一.动机(Motivation) 在软件系统创建过程中,经常面临着“某个对象”的创建工作:由于需求的变化,这个对象(的具体实现)经常面临着剧烈的变化,但是它却拥有比较稳定的接口.如何应对这种变化?如何提供一种“封装机制”来隔离出“这个易变对象”的变化,从而保持系统中“其他依赖对象的对象”不随着需求改变而改变? 二.意图(Intent) 定义一个用于创建对象的接口,让子类决定实例化哪一个类.Factory Method使得一个类的实例化延迟到子类.——<设计模式>GoF 三.结构(Structu

  • .Net设计模式之建造者、生成器模式(Builder)

    一.动机(Motivation) 在软件系统中,有时候面临着“一个复杂对象”的创建工作,其通常由各个部分的子对象用一定的算法构成:由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法却相对稳定.如何应对这种变化?如何提供一种“封装机制”来隔离出“复杂对象的各个部分”的变化,从而保持系统中的“稳定构建算法”不随着需求改变而改变? 二.意图(Intent) 将一个复杂对象的构建与其表示相分离,使得同样的构建过程可以创建不同的表示.——<设计模式>GoF 三.结构(

  • .Net设计模式之原型模式(Prototype)

    一.动机(Motivation) 在软件系统中,经常面临着“某些结构复杂的对象”的创建工作:由于需求的变化,这些对象经常面临着剧烈的变化,但是它们却拥有比较稳定一致的接口.如何应对这种变化?如何向“客户程序(使用这些对象的程序)”隔离出“这些易变对象”,从而使得“依赖这些易变对象的客户程序”不随着需求改变而改变? 二.意图(Intent) 使用原型实例指定创建对象的种类,然后通过拷贝这些原型来创建新的对象——<设计模式>GoF 三.结构(Structure) 我们看了这么多设计模式的类图了,大

  • .NET设计模式之UML类图介绍

    一.概述 UML类图用来定义系统中的类,包括描述类的结构和类之间的关系.类图的主要作用于描述系统的静态结构. 类图的基本模型元素如下: 我们可以看到,一个类图表示为长方形,分为3部分,最上面是类名,中间是类的属性,下面是类的操作.描述信息使用 << >> 包裹. 类图中的实体如下: 1.类名: 正体字说明类是可被实例化的,斜体字说明类为抽象类. 2.属性 如图: 属性一般通过如上的格式表示——<访问权限><属性名>:<属性类型>=<初始值&

  • .Net设计模式之简单工厂模式(Simple Factory)

    简单工厂模式(Simple Factory Pattern)属于类的创建型模式,又叫静态工厂方法模式(Static FactoryMethod Pattern) 是通过专门定义一个类来负责创建其他类的实例,被创建的实例通常都具有共同的父类. 这个模式并不属于GoF23里面的设计模式,其实他属于一个过渡的模式,这个模式是为了引出下一篇要将的模式:工厂模式. 一.UML图 简单工厂模式解决的问题是如何去实例化一个合适的对象. 简单工厂模式的核心思想就是:有一个专门的类来负责创建实例的过程.具体来说,

  • .Net设计模式之抽象工厂模式(Abstract Factory)

    目录 一.动机(Motivation) 二.意图(Intent) 三.结构(Structure) 四.模式的组成 五.抽象工厂的具体代码实现 客户程序: 应用到具体程序(现代风格): 改造 第一种改造 第二种改造 六.抽象工厂的实现要点 抽象工厂模式的优点: 抽象工厂模式的缺点: 抽象工厂模式的使用场景: 七..NET中抽象工厂模式实现 一.动机(Motivation) 在软件系统中,经常面临着“一系列相互依赖的对象”的创建工作:同时,由于需求的变化,往往存在更多系列对象的创建工作. 如何应对这

  • .Net设计模式之单例模式(Singleton)

    一.动机(Motivation) 在软件系统中,经常有这样一些特殊的类,必须保证它们在系统中只存在一个实例,才能确保它们的逻辑正确性.以及良好的效率. 如何绕过常规的构造器,提供一种机制来保证一个类只有一个实例? 这应该是类设计者的责任,而不是使用者的责任. 二.意图(Intent) 保证一个类仅有一个实例,并提供一个该实例的全局访问点 三.结构(Structure) 保证一个类仅有一个实例,并提供一个访问它的全局访问点. 四.单例模式的代码实现 1.单线程Singleton模式的实现 publ

  • Android 单例模式 Singleton 简单实例设计模式解析

    单例模式 Singleton 简单实例设计模式解析 前言 今天我来全面总结一下Android开发中最常用的设计模式 - 单例模式. 关于设计模式的介绍,可以看下我之前写的:1分钟全面了解"设计模式" 目录 1. 引入 1.1 解决的是什么问题 之前说过,设计模式 = 某类特定问题的解决方案,那么单例模式是解决什么问题的解决方案呢? 含义:单例 =一个实例: 解决的问题:降低对象之间的耦合度 解决方法:单例模式,即实现一个类只有一个实例化对象,并提供一个全局访问点 1.2 实例引入 接下

  • Java设计模式单例模式(Singleton)用法解析

    这篇文章主要介绍了Java设计模式单例模式(Singleton)用法详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 单例模式的应用场景: 单例模式(Singleton Pattern)是指确保一个类在任何情况下都绝对只有一个实例.并提供一个全局反访问点.单例模式是创建型模式.单例模式在生活中应用也很广泛,比如公司CEO只有一个,部门经理只有一个等.JAVA中ServletCOntext,ServetContextCOnfig等,还有spri

  • php设计模式之单例模式代码

    php设计模式之单例模式的例子,供大家参考,具体内容如下 <?php /** * php设计模式 单例模式 */ class Fruit{ private static $instanceMap = array(); //protected getter for singleton instances protected static function getSingleton($className){ //保证单例模式 并且不能从控制器实例化和克隆 if (!isset(self::$inst

  • java设计模式之单例模式的详解及优点

    java设计模式之单例模式 定义:如果一个类始终只能创建一个实例,那么这个类被称为单例类,这种设计模式被称为单例模式. Spring框架里面可以将所有生成的bean对象都设置为单例模式,只需要在配置Bean实例时指定scope="singleton"即可,或者不做配置默认即为单例模式. 我们可以创建一个小的Demo来演示单例模式的实现,只需要保证该类只能创建一个实例,我们可以用权限修饰符private修饰该类的构造器. 提供一个创建该类的接口,该接口只能用static修饰,类里面创建一

  • Java设计模式之单例模式实例分析

    本文实例讲述了Java设计模式之单例模式.分享给大家供大家参考,具体如下: 单例模式:(Singleton Pattern)是一个比较简单的模式,其定义如下: Ensure a class has only one instance, and provide a global point of access to it.(确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例) 单例模式,很简单的一个模式.其实在android开发中,很多地方都会用到单例模式,比如某些工具类.Json数

  • java 设计模式之单例模式

    java  设计模式之单例模式 前言: 在软件开发过程中常会有一些对象我们只需要一个,如:线程池(threadpool).缓存(cache).对话框.偏好设置等.这些对象如果制造出多个实例的话可能会导致一些不必要的麻烦,如:程序行为异常.资源使用过量等.这时单例模式就可以确保一个类只有一个实例,并提供全局访问点.下面是从简单的单例类来探讨该用何种方法实现单例模式. /** * 最经典的单例类 */ public class Singleton { // 设置成静态变量来记录Singleton的唯

  • .Net 单例模式(Singleton)

    每台计算机可以有若干个打印机,但只能有一个Printer Spooler, 以避免两个打印作业同时输出到打印机中.每台计算机可以有若干传真卡,但是只应该有一个软件负责管理传真卡,以避免出现两份传真作业同时传到传真卡中的情况.每台计算机可以有若干通信端口,系统应当集中管理这些通信端口,以避免一个通信端口同时被两个请求同时调用. 问题描述: 单例模式 Singleton Pattern 问题解决: (1)单例模式简介: Singleton模式要求一个类有且仅有一个实例,并且提供了一个全局的访问点.这

  • Android编程设计模式之单例模式实例详解

    本文实例讲述了Android编程设计模式之单例模式.分享给大家供大家参考,具体如下: 一.介绍 单例模式是应用最广的模式之一,也可能是很多初级工程师唯一会使用的设计模式.在应用这个模式时,单例对象的类必须保证只有一个实例存在.许多时候整个系统只需要拥有一个全局对象,这样有利于我们协调系统整体的行为. 二.定义 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 三.使用场景 确保某个类有且只有一个对象的场景,避免产生多个对象消耗过多的资源,或者某种类型的对象只应该有且只有一个.例

  • PHP设计模式之单例模式定义与用法分析

    本文实例分析了PHP设计模式之单例模式.分享给大家供大家参考,具体如下: 单例模式(Singleton Pattern 单件模式或单元素模式),是常见的一种设计模式,它有三个特点 1.只能有一个实例 2.必须自行创建这个实例 3.必须给其他对象提供这一实例 下面用PHP代码实现一下 <?PHP /** * Created by PHPStorm. * User: tiansi * Date: 18/1/2 * Time: 下午3:40 */ class Signleton{ private st

随机推荐