欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

c# 单例模式的实现方法

程序员文章站 2022-03-22 16:20:52
单例模式大概是所有设计模式中最简单的一种,如果在面试时被问及熟悉哪些设计模式,你可能第一个答的就是单例模式。单例模式的实现分为两种:饿汉式和懒汉式。前者是在静态构造函数执行时就立即实例化,后者是在程序...

单例模式大概是所有设计模式中最简单的一种,如果在面试时被问及熟悉哪些设计模式,你可能第一个答的就是单例模式。

单例模式的实现分为两种:饿汉式和懒汉式。前者是在静态构造函数执行时就立即实例化,后者是在程序执行过程中第一次需要时再实例化。两者有各自适用的场景,实现方式也都很简单,唯一在设计时要考虑的一个问题就是:实例化时需要保证线程安全。

饿汉式

饿汉式实现很简单,在静态构造函数中立即进行实例化:

public class singleton
{
  private static readonly singleton _instance;
  static singleton()
  {
    _instance = new singleton();
  }

  public static singleton instance
  {
    get
    {
      return _instance;
    }
  }
}

注意,为了确保单例性,需要使用 readonly 关键字声明实例不能被修改。

以上写法可简写为:

public class singleton
{
  private static readonly singleton _instance = new singleton();
  public static singleton instance
  {
    get
    {
      return _instance;
    }
  }
}

这里的 new singleton() 等同于在静态构造函数中实例化。在 c# 7 中还可以进一步简写如下:

public class singleton
{
  public static singleton instance { get; } = new singleton();
}

一句代码就搞定了,此写法,实例化也是在默认的静态构造函数中进行的。如果是饿汉式需求,这种实现是最简单的。有人会问这会不会有线程安全问题,如果多个线程同时调用 singleton.instance 会不会实例化了多个实例。不会,因为 clr 确保了所有静态构造函数都是线程安全的。

注意,不能这么写:

public class singleton
{
  public static singleton instance => new singleton();
}

// 等同于:
public class singleton
{
  public static singleton instance
  {
    get { return new singleton(); }
  }
}

这样会导致每次调用都会创建一个新实例。

懒汉式

懒汉式单例实现需要考虑线程安全问题,先来看一段经典的线程安全的单列模式实现代码:

public sealed class singleton
{
  private static volatile singleton _instance;
  private static readonly object _lockobject = new object();

  public static singleton instance
  {
    get
    {
      if (_instance == null)
      {
        lock (_lockobject)
        {
          if (_instance == null)
          {
            _instance = new singleton();
          }
        }
      }
      return _instance;
    }
  }
}

网上搜索 c# 单例模式,大部分都是这种使用 lock 来确保线程安全的写法,这是经典标准的单例模式的写法,没问题,很放心。在 lock 里外都做一次 instance 空判断,双保险,足以保证线程安全和单例性。但这种写法似乎太麻烦了,而且容易写错。早在 c# 3.5 的时候,就有了更好的写法,使用 lazy<t>。

示例代码:

public class lazysingleton
{
  private static readonly lazy<lazysingleton> _instance =
    new lazy<lazysingleton>(() => new lazysingleton());

  public static lazysingleton instance
  {
    get { return _instance.value; }
  }
}

调用示例:

public class program
{
  public static void main()
  {
    var instance = lazysingleton.instance;
  }
}

使用 lazy<t> 可以使对象的实例化延迟到第一次被调用的时候执行,通过访问它的 value 属性来创建并获取实例,并且读取一个 lazy<t> 实例的 value 属性只会执行一次实例化代码,确保了线程安全。

祝,编码愉快!

以上就是c# 单例模式的实现方法的详细内容,更多关于c# 单例模式的资料请关注其它相关文章!

相关标签: c# 单例模式