c#设计模式之单例模式的实现方式

 更新时间:2020年6月25日 10:35  点击:2194

场景描述

单例模式对于我们来说一点也不模式,是一个常见的名称,单例模式在程序中的实际效果就是:确保一个程序中只有一个实例,并提供一个全局访问点,节省系统资源

单例模式无论是在实际开发中还是在软件应用中比较常见,比如,windows系统的任务管理器、IIS的HttpApplication、实际项目中的日志组件等等

实现方式

单例模式为了实现一个实例,那么只有不把实例创建暴露出去,只通过类本身来创建实例,为了实现效果,需要定义一个私有构造函数

单例模式实现方式有:饿汉式、懒汉式、双重验证式、静态内部类

下面分别对每一种实现方式做一个简单的实例,以及其优缺点

饿汉式

/// <summary>
 /// 创建一个 Singleton 类(饿汉式)
 /// 这种方式比较常用,但容易产生垃圾对象。
 ///优点:没有加锁,执行效率会提高。
 ///缺点:类加载时就初始化,浪费内存。
 ///它基于 classloder 机制避免了多线程的同步问题,不过,instance 在类装载时就实例化,
 ///虽然导致类装载的原因有很多种,在单例模式中大多数都是调用 getInstance 方法,
 ///但是也不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化 instance 显然没有达到 lazy loading 的效果。
 /// </summary>
 public class SingleObject
 {
  //创建 SingleObject 的一个对象
  private static SingleObject instance = new SingleObject();

  //让构造函数为 private,这样该类就不会被实例化
  private SingleObject() {
   Console.WriteLine("我被创建了.饿汉式");
  }

  //获取唯一可用的对象
  public static SingleObject GetInstance()
  {
   return instance;
  }

  public void ShowMessage()
  {
   Console.WriteLine("Hello World.饿汉式");
  }
 }

懒汉式

/// <summary>
 /// 创建一个 Singleton 类(懒汉式)
 /// 这种方式具备很好的 lazy loading,能够在多线程中很好的工作,但是,效率很低,99% 情况下不需要同步。
 /// 优点:第一次调用才初始化,避免内存浪费。
 /// 缺点:懒汉式在单个线程中没有问题,但多个线程同事访问的时候就可能同事创建多个实例,而且这多个实例不是同一个对象。
 /// </summary>
 public class SingleObject1
 {
  //创建 SingleObject 的一个对象
  private static SingleObject1 instance;

  //让构造函数为 private,这样该类就不会被实例化
  private SingleObject1() { }

  //获取唯一可用的对象
  public static SingleObject1 GetInstance()
  {
   if (instance == null)
   {
    instance = new SingleObject1();
    Console.WriteLine("我被创建了.懒汉式");
   }
   return instance;
  }

  public void ShowMessage()
  {
   Console.WriteLine("Hello World.懒汉式");
  }
 }

双重验证式

/// <summary>
 /// 创建一个 Singleton 类(双重验证)
 /// 这种方式具备很好的 lazy loading,能够在多线程中很好的工作,但是,效率很低,99% 情况下不需要同步。
 /// 优点:第一次调用才初始化,避免内存浪费,线程安全。
 /// 缺点:必须加锁 synchronized 才能保证单例,但加锁会影响效率。
 /// </summary>
 public class SingleObject2
 {
  //创建 SingleObject 的一个对象
  private static SingleObject2 instance;

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


  //让构造函数为 private,这样该类就不会被实例化
  private SingleObject2() { }

  //获取唯一可用的对象
  public static SingleObject2 GetInstance()
  {
   //// 如果为空,那么就加锁,创建实例
   if (instance == null)
   {
    lock (locker)
    {
     //// 枷锁成功后,在做一次非空判断,避免在加锁期间以创建了实例而导致重复创建
     if (instance == null)
     {
      instance = new SingleObject2();
      Console.WriteLine("我被创建了.双重验证");
     }
    }
   }
   return instance;
  }

  public void ShowMessage()
  {
   Console.WriteLine("Hello World.双重验证");
  }
 }

静态内部类

/// <summary>
 /// 创建一个 Singleton 类(静态内部类) 
 /// 这种方式不用加锁,在效率上和内存使用上都比较优秀
 /// 克服了饿汉模式的不足饿汉模式执行效率高,由于在类加载的时候初始化导致内存浪费
 /// </summary>
 public class SingletonStatic
 {
  /// <summary>
  /// 内部类
  /// </summary>
  public class SingletonStaticInner
  {
   /// <summary>
   /// 当一个类有静态构造函数时,它的静态成员变量不会被beforefieldinit修饰
   /// 就会确保在被引用的时候才会实例化,而不是程序启动的时候实例化
   /// </summary>
   static SingletonStaticInner() { }

   /// <summary>
   /// 实例化
   /// </summary>
   internal static SingletonStatic singletonStatic = new SingletonStatic();
  }

  /// <summary>
  /// 私有构造函数
  /// </summary>
  private SingletonStatic() {
   Console.WriteLine("我被创建了.静态内部类");
  }

  /// <summary>
  /// 获取实例
  /// </summary>
  /// <returns></returns>
  public static SingletonStatic GetInstance()
  {
   return SingletonStaticInner.singletonStatic;
  }

  public void ShowMessage()
  {
   Console.WriteLine("Hello World.静态内部类");
  }
 }

每一种创建方式测试

创建一个控制台程序,通过多线程对每一种实现方式使用,查看其实例次数分析:

/*
 介绍
意图:保证一个类仅有一个实例,并提供一个访问它的全局访问点。
主要解决:一个全局使用的类频繁地创建与销毁。
何时使用:当您想控制实例数目,节省系统资源的时候。
如何解决:判断系统是否已经有这个单例,如果有则返回,如果没有则创建。
关键代码:构造函数是私有的。
应用实例:
典型的已有应用:
1、windows的任务管理器等
2、IIS的HttpApplication,所有的HttpModule都共享一个HttpApplication实例
在项目中的实际使用场景:
1、日志组件
2、多线程线程池管理
3、网站计数器
4、配置文件管理
  */
 
class Program
 {
  static void Main(string[] args)
  {
   TaskFactory taskFactory = new TaskFactory();
   List<Task> taskList = new List<Task>();

   //// 测试--饿汉式 
   for (int i = 0; i < 5; i++)
   {
    taskList.Add(taskFactory.StartNew(() =>
    {
     SingleObject.GetInstance();
    }));
   }

   //// 测试--懒汉式 
   for (int i = 0; i < 5; i++)
   {
    taskList.Add(taskFactory.StartNew(() =>
    {
     SingleObject1.GetInstance();
    }));
   }

   //// 测试--双重验证 
   for (int i = 0; i < 5; i++)
   {
    taskList.Add(taskFactory.StartNew(() =>
    {
     SingleObject2.GetInstance();
    }));
   }

   //// 测试--静态内部类 
   for (int i = 0; i < 5; i++)
   {
    taskList.Add(taskFactory.StartNew(() =>
    {
     SingletonStatic.GetInstance();
    }));
   }

   Console.ReadLine();
  }
 }

运行结果:

  

通过结果可以看出:懒汉式实际创建了2个实例,所以在多线程中,懒汉式有线程不安全问题

总结

根据单例模式是每一种实现方式对比分析,在实际使用过程中:

如果是单线程应用环境,建议可以采用懒汉模

如果是多线程应用环境,建议采用静态内部类方式

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对猪先飞的支持。

[!--infotagslink--]

相关文章

  • C#实现简单的登录界面

    我们在使用C#做项目的时候,基本上都需要制作登录界面,那么今天我们就来一步步看看,如果简单的实现登录界面呢,本文给出2个例子,由简入难,希望大家能够喜欢。...2020-06-25
  • 浅谈C# 字段和属性

    这篇文章主要介绍了C# 字段和属性的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下...2020-11-03
  • C#中截取字符串的的基本方法详解

    这篇文章主要介绍了C#中截取字符串的的基本方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-11-03
  • C#实现简单的Http请求实例

    这篇文章主要介绍了C#实现简单的Http请求的方法,以实例形式较为详细的分析了C#实现Http请求的具体方法,需要的朋友可以参考下...2020-06-25
  • C#连接SQL数据库和查询数据功能的操作技巧

    本文给大家分享C#连接SQL数据库和查询数据功能的操作技巧,本文通过图文并茂的形式给大家介绍的非常详细,需要的朋友参考下吧...2021-05-17
  • C#中new的几种用法详解

    本文主要介绍了C#中new的几种用法,具有很好的参考价值,下面跟着小编一起来看下吧...2020-06-25
  • 使用Visual Studio2019创建C#项目(窗体应用程序、控制台应用程序、Web应用程序)

    这篇文章主要介绍了使用Visual Studio2019创建C#项目(窗体应用程序、控制台应用程序、Web应用程序),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
  • C#开发Windows窗体应用程序的简单操作步骤

    这篇文章主要介绍了C#开发Windows窗体应用程序的简单操作步骤,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-04-12
  • C#从数据库读取图片并保存的两种方法

    这篇文章主要介绍了C#从数据库读取图片并保存的方法,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下...2021-01-16
  • C#和JavaScript实现交互的方法

    最近做一个小项目不可避免的需要前端脚本与后台进行交互。由于是在asp.net中实现,故问题演化成asp.net中jiavascript与后台c#如何进行交互。...2020-06-25
  • C++调用C#的DLL程序实现方法

    本文通过例子,讲述了C++调用C#的DLL程序的方法,作出了以下总结,下面就让我们一起来学习吧。...2020-06-25
  • 轻松学习C#的基础入门

    轻松学习C#的基础入门,了解C#最基本的知识点,C#是一种简洁的,类型安全的一种完全面向对象的开发语言,是Microsoft专门基于.NET Framework平台开发的而量身定做的高级程序设计语言,需要的朋友可以参考下...2020-06-25
  • C#变量命名规则小结

    本文主要介绍了C#变量命名规则小结,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-09-09
  • C#绘制曲线图的方法

    这篇文章主要介绍了C#绘制曲线图的方法,以完整实例形式较为详细的分析了C#进行曲线绘制的具体步骤与相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下...2020-06-25
  • C# 中如何取绝对值函数

    本文主要介绍了C# 中取绝对值的函数。具有很好的参考价值。下面跟着小编一起来看下吧...2020-06-25
  • c#自带缓存使用方法 c#移除清理缓存

    这篇文章主要介绍了c#自带缓存使用方法,包括获取数据缓存、设置数据缓存、移除指定数据缓存等方法,需要的朋友可以参考下...2020-06-25
  • c#中(&&,||)与(&,|)的区别详解

    这篇文章主要介绍了c#中(&&,||)与(&,|)的区别详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-06-25
  • 经典实例讲解C#递归算法

    这篇文章主要用实例讲解C#递归算法的概念以及用法,文中代码非常详细,帮助大家更好的参考和学习,感兴趣的朋友可以了解下...2020-06-25
  • C#学习笔记- 随机函数Random()的用法详解

    下面小编就为大家带来一篇C#学习笔记- 随机函数Random()的用法详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
  • C#中list用法实例

    这篇文章主要介绍了C#中list用法,结合实例形式分析了C#中list排序、运算、转换等常见操作技巧,具有一定参考借鉴价值,需要的朋友可以参考下...2020-06-25