浅谈C#单例模式的实现和性能对比

发布时间 - 2026-01-11 03:23:54    点击率:

简介

单例指的是只能存在一个实例的类(在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<T>类型

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

  public static Singleton Instance 
  {
    get 
    {
      return lazy.Value;
    }
  }

  private Singleton()
  {
  }
}

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

性能差异

之前的实现中,我们都在强调代码的线程安全性和延迟加载。然而在实际使用中,如果你的单例类的初始化不是一个很耗时的操作或者初始化顺序不会导致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#单例模式  # C#实现单例模式  # c# 单例模式的实现  # C#实现单例模式的几种方法总结  # c# 单例模式的实现方法  # c#设计模式之单例模式的实现方式  # c#单例模式(Singleton)的6种实现  # C#窗口实现单例模式的方法  # C#实现单例模式的多种方式  # 会在  # 加载  # 几种  # 死锁  # 它是  # 自己的  # 来实现  # 也有  # 上了  # 有一个  # 很容易  # 只会  # 仍然是  # 创建一个  # 新建一个  # 类中  # 的是  # 是一个  # 几个  # 就会 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Python文件操作最佳实践_稳定性说明【指导】  做企业网站制作流程,企业网站制作基本流程有哪些?  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  如何在宝塔面板中创建新站点?  Laravel如何与Docker(Sail)协同开发?(环境搭建教程)  Laravel怎么使用artisan命令缓存配置和视图  php增删改查怎么学_零基础入门php数据库操作必知基础【教程】  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  laravel怎么通过契约(Contracts)编程_laravel契约(Contracts)编程方法  北京的网站制作公司有哪些,哪个视频网站最好?  Laravel如何实现API资源集合?(Resource Collection教程)  Android仿QQ列表左滑删除操作  Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全  Laravel DB事务怎么使用_Laravel数据库事务回滚操作  JavaScript中如何操作剪贴板_ClipboardAPI怎么用  南京网站制作费用,南京远驱官方网站?  焦点电影公司作品,电影焦点结局是什么?  矢量图网站制作软件,用千图网的一张矢量图做公司app首页,该网站并未说明版权等问题,这样做算不算侵权?应该如何解决?  头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?  Python自然语言搜索引擎项目教程_倒排索引查询优化案例  如何在IIS服务器上快速部署高效网站?  轻松掌握MySQL函数中的last_insert_id()  Laravel如何与Vue.js集成_Laravel + Vue前后端分离项目搭建指南  如何在HTML表单中获取用户输入并用JavaScript动态控制复利计算循环  Laravel如何使用Livewire构建动态组件?(入门代码)  Laravel如何实现一对一模型关联?(Eloquent示例)  如何用虚拟主机快速搭建网站?详细步骤解析  高端智能建站公司优选:品牌定制与SEO优化一站式服务  如何选择PHP开源工具快速搭建网站?  如何利用DOS批处理实现定时关机操作详解  Laravel Artisan命令怎么自定义_创建自己的Laravel命令行工具完全指南  用yum安装MySQLdb模块的步骤方法  如何在万网主机上快速搭建网站?  Laravel项目怎么部署到Linux_Laravel Nginx配置详解  Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】  Laravel怎么调用外部API_Laravel Http Client客户端使用  Laravel如何设置定时任务(Cron Job)_Laravel调度器与任务计划配置  电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?  jQuery validate插件功能与用法详解  怎么用AI帮你为初创公司进行市场定位分析?  在Oracle关闭情况下如何修改spfile的参数  Laravel的Blade指令怎么自定义_创建你自己的Laravel Blade Directives  创业网站制作流程,创业网站可靠吗?  佛山网站制作系统,佛山企业变更地址网上办理步骤?  高防网站服务器:DDoS防御与BGP线路的AI智能防护方案  Laravel如何实现用户密码重置功能?(完整流程代码)  香港服务器WordPress建站指南:SEO优化与高效部署策略  Laravel任务队列怎么用_Laravel Queues异步处理任务提升应用性能  Python制作简易注册登录系统  javascript基本数据类型及类型检测常用方法小结