C#单例模式的实现以及性能对比的实例

这篇文章主要介绍了浅谈c#单例模式的实现和性能对比的相关资料,详细的介绍了6种实现方式,需要的朋友可以参考下

简介

单例指的是只能存在一个实例的类(在C#中,更准确的说法是在每个AppDomain之中只能存在一个实例的类,它是软件工程中使用最多的几种模式之一。在第一个使用者创建了这个类的实例之后,其后需要使用这个类的就只能使用之前创建的实例,无法再创建一个新的实例。通常情况下,单例会在第一次被使用时创建。本文会对C#中几种单例的实现方式进行介绍,并分析它们之间的线程安全性和性能差异。

单例的实现方式有很多种,但从最简单的实现(非延迟加载,非线程安全,效率低下),到可延迟加载,线程安全,且高效的实现,它们都有一些基本的共同点:

单例类都只有一个private的无参构造函数

类声明为sealed(不是必须的)

类中有一个静态变量保存着所创建的实例的引用

单例类会提供一个静态方法或属性来返回创建的实例的引用(eg.GetInstance)

几种实现

一非线程安全

//Bad code! Do not use!public sealed class Singleton{  private static Singleton instance = null;  private Singleton()  {  }  public static Singleton instance  {    get    {      if (instance == null)      {        instance = new Singleton();      }      return instance;    }  }}

这种方法不是线程安全的,会存在两个线程同时执行if (instance == null)并且创建两个不同的instance,后创建的会替换掉新创建的,导致之前拿到的reference为空。

二简单的线程安全实现

public sealed class Singleton{  private static Singleton instance = null;  private static readonly object padlock = new object();  Singleton()  {  }  public static Singleton Instance  {    get    {      lock (padlock)      {        if (instance == null)        {          instance = new Singleton();        }        return instance;      }    }  }}

相比较于实现一,这个版本加上了一个对instance的锁,在调用instance之前要先对padlock上锁,这样就避免了实现一中的线程冲突,该实现自始至终只会创建一个instance了。但是,由于每次调用Instance都会使用到锁,而调用锁的开销较大,这个实现会有一定的性能损失。

注意这里我们使用的是新建一个private的object实例padlock来实现锁操作,而不是直接对Singleton进行上锁。直接对类型上锁会出现潜在的风险,因为这个类型是public的,所以理论上它会在任何code里调用,直接对它上锁会导致性能问题,甚至会出现死锁情况。

Note: C#中,同一个线程是可以对一个object进行多次上锁的,但是不同线程之间如果同时上锁,就可能会出现线程等待,或者严重的会出现死锁情况。因此,我们在使用lock时,尽量选择类中的私有变量上锁,这样可以避免上述情况发生。

三双重验证的线程安全实现

public sealed calss Singleton{  private static Singleton instance = null;  private static readonly object padlock = new object();  Singleton()  {  }  public static Singleton Instance  {    get    {      if (instance == null)      {        lock (padlock)        {          if (instance == null)          {            instance = new Singleton();          }        }      }      return instance;    }  } }

在保证线程安全的同时,这个实现还避免了每次调用Instance都进行lock操作,这会节约一定的时间。

但是,这种实现也有它的缺点:

1无法在Java中工作。(具体原因可以见原文,这边没怎么理解)

2程序员在自己实现时很容易出错。如果对这个模式的代码进行自己的修改,要倍加小心,因为double check的逻辑较为复杂,很容易出现思考不周而出错的情况。

四不用锁的线程安全实现

public sealed class Singleton{  //在Singleton第一次被调用时会执行instance的初始化  private static readonly Singleton instance = new Singleton();  //Explicit static consturctor to tell C# compiler   //not to mark type as beforefieldinit  static Singleton()  {  }  private Singleton()  {  }  public static Singleton Instance  {    get    {      return instance;    }  }}

这个实现很简单,并没有用到锁,但是它仍然是线程安全的。这里使用了一个static,readonly的Singleton实例,它会在Singleton第一次被调用的时候新建一个instance,这里新建时候的线程安全保障是由.NET直接控制的,我们可以认为它是一个原子操作,并且在一个AppDomaing中它只会被创建一次。

这种实现也有一些缺点:

1instance被创建的时机不明,任何对Singleton的调用都会提前创建instance
2static构造函数的循环调用。如有A,B两个类,A的静态构造函数中调用了B,而B的静态构造函数中又调用了A,这两个就会形成一个循环调用,严重的会导致程序崩溃。
3我们需要手动添加Singleton的静态构造函数来确保Singleton类型不会被自动加上beforefieldinit这个Attribute,以此来确保instance会在第一次调用Singleton时才被创建。
4readonly的属性无法在运行时改变,如果我们需要在程序运行时dispose这个instance再重新创建一个新的instance,这种实现方法就无法满足。

五完全延迟加载实现(fully lazy instantiation)

public sealed class Singleton{  private Singleton()  {  }  public static Singleton Instance   {    get    {      return Nested.instance;    }  }  private class Nested  {    // Explicit static constructor to tell C# compiler    // not to mark type as beforefieldinit    static Nested()    {    }    internal static readonly Singleton instance = new Singleton();  }}

实现五是实现四的包装。它确保了instance只会在Instance的get方法里面调用,且只会在第一次调用前初始化。它是实现四的确保延迟加载的版本。

六 使用.NET4的Lazy类型

public sealed class Singleton{  private static readonly Lazy lazy = new Lazy(() => new Singleton());  public static Singleton Instance   {    get     {      return lazy.Value;    }  }  private Singleton()  {  }}

.NET4或以上的版本支持Lazy来实现延迟加载,它用最简洁的代码保证了单例的线程安全和延迟加载特性。

性能差异

之前的实现中,我们都在强调代码的线程安全性和延迟加载。然而在实际使用中,如果你的单例类的初始化不是一个很耗时的操作或者初始化顺序不会导致bug,延迟初始化是一个可有可无的特性,因为初始化所占用的时间是可以忽略不计的。

在实际使用场景中,如果你的单例实例会被频繁得调用(如在一个循环中),那么为了保证线程安全而带来的性能消耗是更值得关注的地方。

为了比较这几种实现的性能,我做了一个小测试,循环拿这些实现中的单例9亿次,每次调用instance的方法执行一个count++操作,每隔一百万输出一次,运行环境是MBP上的Visual Studio for Mac。结果如下:

线程安全性 延迟加载 测试运行时间(ms)

实现一否是15532实现二是是45803实现三是是15953实现四是不完全14572实现五是是14295实现六是是22875

测试方法并不严谨,但是仍然可以看出,方法二由于每次都需要调用lock,是最耗时的,几乎是其他几个的三倍。排第二的则是使用.NET Lazy类型的实现,比其他多了二分之一左右。其余的四个,则没有明显区别。

总结

总体来说,上面说的多种单例实现方式在现今的计算机性能下差距都不大,除非你需要特别大并发量的调用instance,才会需要去考虑锁的性能问题。

对于一般的开发者来说,使用方法二或者方法六来实现单例已经是足够好的了,方法四和五则需要对C#运行流程有一个较好的认识,并且实现时需要掌握一定技巧,并且他们节省的时间仍然是有限的。

引用

本文大部分是翻译自Implementing the Singleton Pattern in C#,加上了一部分自己的理解。这是我搜索static readonly field initializer vs static constructor initialization时看到的,在这里对两位作者表示感谢。

以上就是C#单例模式的实现以及性能对比的实例的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432679.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 08:30:03
下一篇 2025年12月17日 08:30:16

相关推荐

  • .NET支付宝App支付接入的实例分析

    一、前言        最近也是为了新产品忙得起飞,博客都更新的慢了。新产品为了方便用户支付,需要支付宝扫码接入。这活落到了我的身上。产品是Windows系统下的桌面软件,通过软件生成二维码支付。界面以原生的MVVM编写,下面叙述一下基本的过程,做过的老司机可以直接点关闭了。 二、申请接口     …

    好文分享 2025年12月17日
    000
  • .Net Core之实现下载文件的实例

    本篇将和大家分享的丝.netcore下载文件,常见的下载有两种:a标签直接指向下载文件地址和post或get请求后台输出文件流的方式,本篇也将围绕这两种来分享;如果对您有好的帮助,请多多支持。 允许站点不识别content-type下载文件(即:不受mime类型限制下载) 如何允许下载.nupkg和…

    2025年12月17日 好文分享
    000
  • C# ArrayListd的长度问题解决

    c# arraylistd的长度问题解决 namespace ArrayListd的长度问题{ class Program { static void Main(string[] args) { //需要的参数是object类型 //alt+shift+F10添加引用using System.Col…

    好文分享 2025年12月17日
    000
  • C# 加密类工具实例分析

    5.SHA1加密 //sha1加密 public static String getSha1(String str){ if(str==null||str.length()==0){ return null; } char hexDigits[] = {‘0′,’1′,’2′,’3′,’4’,’5’…

    好文分享 2025年12月17日
    000
  • C#实现杨辉三角的示例

    这篇文章主要介绍了c# 中杨辉三角的实现的相关资料,希望通过本文大家能掌握这部分内容,需要的朋友可以参考下 C# 中杨辉三角的实现 问题描述:创建一个程序来求三角形。该程序提示用户输入数据,然后显示出杨辉三角的规律。            // 输入描述:杨辉三角长,代表数值            …

    好文分享 2025年12月17日
    000
  • C#编写Windows服务程序的图文详解

    本文介绍了如何用c#创建、安装、启动、监控、卸载简单的windows service 的内容步骤和注意事项,需要的朋友可以参考下 一、创建一个Windows Service 1)创建Windows Service项目   2)对Service重命名 将Service1重命名为你服务名称,这里我们命名…

    2025年12月17日 好文分享
    000
  • C#中关于foreach实现的原理详解

    这篇文章主要为大家详细介绍了c#中foreach实现原理,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 本文主要记录我在学习C#中foreach遍历原理的心得体会。 对集合中的要素进行遍历是所有编码中经常涉及到的操作,因此大部分编程语言都把此过程写进了语法中,比如C#中的foreach。经常会看…

    好文分享 2025年12月17日
    000
  • C#通过KD树进行距离最近点的查找的实例分析

    这篇文章主要为大家详细介绍了c#通过kd树进行距离最近点的查找,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 本文首先介绍Kd-Tree的构造方法,然后介绍Kd-Tree的搜索流程及代码实现,最后给出本人利用C#语言实现的二维KD树代码。这也是我自己动手实现的第一个树形的数据结构。理解上难免会有…

    好文分享 2025年12月17日
    000
  • c#如何生成二维码的示例分享

    引用zxing类库 实现功能: 1生成带有Logo二维码 2 将二维码绘制到图片上 3 图片上绘制文字  生成二维码 public string CreateQrCode(string md5Str,string name,int sex) { string str = sex == 1? “先生”…

    好文分享 2025年12月17日
    000
  • C# 数组作为参数传递出现的问题解决

    原则:尽可能控制对数据的修改,如果可以预测某个数据不会或不应该被改变,就要对其控制,而不要期望使用这个数据的调用者不会改变其值。 如果参数在使用过程中被意外修改,将会带来不可预知的结果,而且这种错误很难被检查到,所以我们在设计方法参数的时候,要充分考虑传递引用类型参数或者引用方式传递引用类型参数可能…

    好文分享 2025年12月17日
    000
  • .net core使用Redis发布订阅方法介绍

    本篇文章主要介绍了.net core如何使用redis发布订阅,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧 Redis是一个性能非常强劲的内存数据库,它一般是作为缓存来使用,但是他不仅仅可以用来作为缓存,比如著名的分布式框架dubbo就可以用Redis来做服务注册中心…

    2025年12月17日 好文分享
    000
  • 详解.Net Core使用Socket与树莓派进行通信

    这篇文章主要为大家详细介绍了.net core使用socket与树莓派进行通信,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 前言 去年买的树莓派一直放在抽屉里吃灰,前些阵子Debian 9发布,也不出意外的支持了树莓派。 于是重新拿出读卡器又重新了装上了Debian桌面版系统。 介绍 现在这个…

    2025年12月17日
    000
  • C#根据表格偶数与奇数加载不同颜色的实例分析

    这篇文章主要介绍了c# 根据表格偶数、奇数加载不同颜色,需要的朋友可以参考下 效果图:        //偶数随机 Random evenRanm = new Random(); //奇数随机 Random oddRanm = new Random(); string[] listColor = n…

    2025年12月17日
    000
  • C#之FastSocket实战项目的示例分享

    一、FastSocket课程介绍  .net框架虽然微软提供了socket通信的类库,但是还有很多事情要自己处理,比如tcp协议需要处理分包、组包、粘包、维护连接列表等,udp协议需要处理丢包、乱序,而且对于多连接并发,还要自己处理多线程等等。本期分享课程阿笨给大家带来的是来源于github开源so…

    2025年12月17日 好文分享
    000
  • c#之浮点数计算问题的解决

    给大家看个计算题,看看大家的算术能力。 0.1 +0.1 +0.1 – 0.3 等于几? 大家可能会说这么简单的问题,是不是看不起我?肯定等于0啊。 如果大家直接算的是没有问题的,但是如果用计算机呢? 见证奇迹的时刻到了,看代码: void Main(){ var f = 0.1 +0.…

    2025年12月17日
    000
  • C#使用AForge实现摄像头录像功能的案例

    这篇文章主要介绍了c#调用aforge实现摄像头录像的示例代码,非常具有实用价值,需要的朋友可以参考下 1:首先下载库文件>> 也可以去官网寻找>> 下载本教程全代码>> 输出为MP4需要用到ffmpeg相关的文件,我打包的库已经带了,去官网找的库可以在这个目录找…

    2025年12月17日 好文分享
    000
  • C#中关于ActiveMQ的应用详解

    activemq是个好东东,不必多说。activemq提供多种语言支持,如java, c, c++, c#, ruby, perl, python, php等。由于我在windows下开发gui,比较关心c++和c#,其中c#的activemq很简单,apache提供nms(.net messagi…

    好文分享 2025年12月17日
    000
  • .net中关于异步性能测试的示例代码

    很久没有写博客了,今年做的产品公司这两天刚刚开了发布会,稍微清闲下来,想想我们做的产品还有没有性能优化空间,于是想到了.net的异步可以优化性能,但到底能够提升多大的比例呢?恰好有一个朋友正在做各种语言的异步性能测试(有关异步和同步的问题,请参考客《aio与bio接口性能对比》),于是我今天写了一个…

    2025年12月17日 好文分享
    000
  • .NET Core中遇到的一些坑的图文详解

     最近.net core升级到2.0后开始慢慢捣鼓的多了起来,但遇到了不少坑,所以特来记录下。 第一个坑  条件编译符   我们在编写一些方法的时候通常会为Debug模式增加一些输出日志等以便我们检查,也会为Release模式增加或修改一些特定的参数,但今天我在写这些的时候就遇到了这个坑#if !D…

    2025年12月17日 好文分享
    000
  • C#实现表格隔行换色

    这篇文章主要介绍了c# 根据表格偶数、奇数加载不同颜色,需要的朋友可以参考下 效果图:        //偶数随机 Random evenRanm = new Random(); //奇数随机 Random oddRanm = new Random(); string[] listColor = n…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信