C#单例模式(Singleton Pattern)详解

 更新时间:2017年08月28日 08:35:55   作者:SuagrMatl  
这篇文章主要为大家详细介绍了C#单例模式Singleton Pattern的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

(新手写博客,主要是对自己学习的归纳总结。会对很多小细节详解。)

单例模式的定义:

确保一个类只有一个实例,并提供一个全局访问点。

首先实例大家应该都明白就是类生成对象的过程简单的就是String s=new String(),则s就是个实例。

Q:如何只生成一个实例?

A:1)首先必须将构造函数变为私有从而防止其他类实例化,并且只能有一个构造函数。因为系统会默认一个无参构造函数,而且默认public访问修饰符。 所以必须写一个私有无参让默认无效。(通常单例模式都是不带形参的)

   2)在该类中声明一个自己本身的静态实例,然后通过静态方法返回。

Q:如何提供一个全局访问点?

A:在类中创建一个公共并且静态的属性。(因为静态方法是类中的一个成员方法,属于整个类,即不用创建任何对象也可以直接调用。单例模式是不允许其他类实例的。)

代码:

分为两种模式:

1.LAZY模式

就是延迟加载, 设计模式是为了避免一些无谓的性能开销而提出来的,所谓延迟加载就是当在真正需要数据(读取属性值)的时候,才真正执行数据加载操作.有效使用它可以大大提高系统性能。

2.饿汉模式

与LAZY模式相反 ,加载时会将自己实例化。起来最容易的单例模式。

分析代码1:(经典)

// 不要用这种方式
public sealed class Singleton
{
  private static Singleton instance=null;//声明自己本身的静态实例
  private Singleton(){}//私有构造
  public static Singleton Instance() //提供全局访问点
  {
      if (instance==null)//实例不存在则创建
      {
        instance = new Singleton();
      }
      return instance;
  }
}

该代码仅供理解,单例模式的定义。

 问题:该方法是非线程安全的,当有两个线程同时进入时,如果instance为null则都会创建实例。实际上,在测试以前,实例就已经有可能被创建了,但是内存模型不能保证这个实例能被其他的线程看到。

下面我们优化改进

分析代码2:(非安全线程)

public sealed class Singleton
{
  private static Singleton instance = null;
  private static readonly object padlock = new object();//定义一个标识确保线程同步
  Singleton(){}
  public static Singleton Instance()
  {
      lock (padlock)//线程到达时加锁 运行完之后解锁 当遇到加锁线程就会挂起等待解锁
      {
        if (instance == null)
        {
          instance = new Singleton();
        }
        return instance;
      }
  }
}

以上解决了多线程问题。

问题:性能上来说,锁变成了每次都必须的当这个实例被响应的时候。此时完全没必要对线程辅助对象加锁之后再去判断,所以上面的实现方式增加了额外的开销。

下面我们进行优化改进:

 代码分析3:(双重锁定)

public sealed class Singleton
{
  private static Singleton instance = null;
  private static readonly object padlock = new object();
  Singleton(){}
  public static Singleton Instance
  {
    get
    {
      if (instance == null)//外层的if语句块,这使得每个线程欲获取实例时不必每次都得加锁,因为只有实例为空时(即需要创建一个实例),才需加锁创建
      {
        lock (padlock)
        {
          if (instance == null)
          {
            instance = new Singleton();
          }
        }
      }
      return instance;
    }
  }
}

这种“双重检查锁定”理论上是完美的

问题是:并不能保证它会在单处理器或多处理器计算机上顺利运行。(反正就是有问题吧 之后再研读一下 看看具体是怎么回事)

代码分析4:(不完全LAZY)

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

  // 显示的static 构造函数
  //静态构造函数抑制了beforefieldinit 特性(访问成员之前就执行静态函数)
  static Singleton(){}
  private Singleton(){}
  public static Singleton Instance
  {
    get
    {
      return instance;
    }
  }
}

不完全LAZY模式(通过抑制beforefildinit特性并不能起到太大的效果)

 代码分析5:(完全LAZY)

public sealed class Singleton
{
  private Singleton(){}
  public static Singleton Instance { get { return Nested.instance; }}   
//嵌套类
  private class Nested
  {
    //抑制beforefieldinit特性
    static Nested(){}
    internal static readonly Singleton instance = new Singleton();
  }
}

这里使用了嵌套类(嵌套类型是LAZY加载的,也就是说嵌套类型在使用他时才会初始化)

代码分析6:(Lazy<T>)

public sealed class Singleton
{
//使用.NET4 Lazy<T>
  private static readonly Lazy<Singleton> lazy =new Lazy<Singleton>(() => new Singleton());
  public static Singleton Instance { get { return lazy.Value; } }
  private Singleton() {}
}

Lazy<T> 对象初始化默认是线程安全的,在多线程环境下,第一个访问 Lazy<T> 对象的 Value 属性的线程将初始化 Lazy<T> 对象,以后访问的线程都将使用第一次初始化的数据。

以上全部是LAZY模式,现在了解下饿汉模式

代码分析7:

public sealed class Singleton
{
    private static readonly Singleton instance=new Singleton();//直接实例化
    private Singleton(){}
    public static Singleton Instance()
    {
        return instance;
    }
}

在这种模式下,无需自己解决线程安全性问题,CLR会给我们解决。由此可以看到这个类被加载时,会自动实例化这个类,而不用在第一次调用Instance()后才实例化出唯一的单例对象。

为了优化系统当然还是选择优化模式。LAZY模式最好的应该是使用Lazy<T>简短安全。

以上都是我在博客园对单例模式学习的总结。今后会补充一些具体的项目中的案例让和我一样的新手更容易吸收理解最终达到举一反三。这里重点参考了这里的文章,希望大家有指教的地方多多提点。感谢。

相关文章

  • .net使用Aspose.Words进行Word替换操作的实现代码

    .net使用Aspose.Words进行Word替换操作的实现代码

    之前在工作中,需要实现Word打印功能,并且插入图片。当时采取的方式则是使用书签进行操作。首先在word内插入书签,完成后,存为模板。程序加载该模板,找到书签,并在指定位置写入文字即可
    2013-05-05
  • C#中变量、常量、枚举、预处理器指令知多少

    C#中变量、常量、枚举、预处理器指令知多少

    这篇文章主要介绍了c#共有其中变量类型有:静态变量、实类变量、数组元素、数值参数、引用参数、输出参数和局部变量,需要的朋友可以参考一下
    2017-04-04
  • 详解C#实现MD5加密的示例代码

    详解C#实现MD5加密的示例代码

    本篇文章主要介绍了C#实现MD5加密的示例代码,详细的介绍了几种方法,具有一定的参考价值,有兴趣的可以了解一下。
    2016-12-12
  • C#编程之事务用法

    C#编程之事务用法

    这篇文章主要介绍了C#编程之事务用法,结合实例形式对比分析了C#中事务提交与回滚的具体实现技巧与相关注意事项,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-11-11
  • C#复合模式(Composite Pattern)实例教程

    C#复合模式(Composite Pattern)实例教程

    这篇文章主要介绍了C#复合模式(Composite Pattern),以实例形式讲述了复合模式在树形结构中的应用,需要的朋友可以参考下
    2014-09-09
  • c#设计模式 适配器模式详细介绍

    c#设计模式 适配器模式详细介绍

    结构模式(Structural Pattern)描述如何将类或者对象结合在一起形成更大的结构。结构模式描述两种不同的东西:类与类的实例。根据这一点,结构模式可以分为类的结构模式和对象的结构模式
    2012-10-10
  • 轻松学习C#的装箱与拆箱

    轻松学习C#的装箱与拆箱

    轻松学习C#的装箱与拆箱,在之前的文章简单的提到了轻松学习C#的装箱与拆箱,本文带着大家更加详细的介绍轻松学习C#的装箱与拆箱,感兴趣的小伙伴们可以参考一下
    2015-11-11
  • C#根据Word模版生成Word文件

    C#根据Word模版生成Word文件

    这篇文章主要为大家详细介绍了C#根据Word模版生成Word文件的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • 聊聊Unity 自定义日志保存的问题

    聊聊Unity 自定义日志保存的问题

    这篇文章主要介绍了Unity 自定义日志保存的问题,之前unity5.x在代码中写了debug.log打包之后在当前程序文件夹下会有个对应的"outlog.txt",后来进行了更改,今天通过代码给大家介绍了Unity日志保存的问题,需要的朋友一起看看吧
    2021-05-05
  • C#使用stackalloc分配堆栈内存和非托管类型详解

    C#使用stackalloc分配堆栈内存和非托管类型详解

    这篇文章主要为大家介绍了C#使用stackalloc分配堆栈内存和非托管类型详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪<BR>
    2022-12-12

最新评论