一、单例模式优点

  1. 单例模式核心在于对于某个单例类,在系统中同时只存在唯一一个实例,并且该实例容易被外界所访问;
  2. 意味着在内存中,只存在一个实例,减少了内存开销;

二、单例模式特点

  1. 只存在唯一一个实例;
  2. 提供统一对外访问接口,使得全局可对该单例的唯一实例进行访问;
  3. 自行实例化(私有构造函数,不允许外界对其进行实例化)。

三、单例模式使用

  1. 资源管理器,资源对象数据的加载和卸载(无状态不需要实例化的对象);
  2. 单一客户端连接服务器等;
  3. 生命周期在游戏中永不消毁的对象。

四、单例模式注意点

  1. 注意线程安全问题,在多线程、高并发的情况下,可能同时产生多个实例,违背了单例模式。
  2. Unity中如果过度使用单例模式,将会导致代码耦合度非常高,脚本与脚本之间的耦合,代码的后续拓展变得非常麻烦。一个过分依赖单例模式的开发者不能成为一个好的开发者,也不会去接触到更多优秀的设计模式。个人推荐ECS 实体 - 组件式编程。
  3. Unity中暂时不需要考虑多线程问题,Unity就只有一个主线程和开启多个辅助协程,不会出现多线程并发问题。
  4. 控制游戏对象的生成和销毁并不建议使用单例模式,可通过主游戏逻辑InGame进行事件下发,自行管理Update,使用工厂来进行对象的创建和销毁。

五、单例模式常见模式

  1. 懒汉模式(最常用)

    1.1 提供私有构造函数;

    1.2 自行实例化;

    1.3 提供唯一实例,并且对外提供全局静态访问接口对该实例进行访问;

    • 代码如下:

      /// <summary>
      /// 普通模式
      /// </summary>
      public class Singleton
      {
      private static Singleton _instance = null; private Singleton()
      {
      } public static Singleton GetInstance()
      {
      if (_instance == null)
      {
      _instance = new Singleton();
      }
      return _instance;
      }
      }
  2. 饿汉模式

    2.1 本类内部预先自行实例化出唯一实例;

    2.2 对外提供唯一访问接口(静态方法),对预先实例化的唯一实例进行访问;

    2.3 私有构造函数;

    • 代码如下:
         /// <summary>
    /// 饿汉单例模式
    /// </summary>
    public class Singleton
    {
    // 自行预先实例化,内部定义自己唯一实例,只供内部使用 //
    private readonly static Singleton Instance = new Singleton(); private Singleton()
    {
    // Do Something
    } // 提供外部访问的静态方法,来对内部唯一实例进行访问 //
    public static Singleton GetInstance()
    {
    return Instance;
    }
    }
  3. 双重锁模式(解决线程安全问题)

    3.1 保证多线程中只存在唯一实例

    • 代码如下:

      /// <summary>
      /// 双重锁单例模式
      /// </summary>
      public class Singleton
      {
      private static Singleton _instance = null;
      private static readonly object _syslock = new object();
      private Singleton()
      {
      } public static Singleton GetInstance()
      {
      // 最开始判断不存在的时候,该类从来未被实例化过 //
      if (_instance == null)
      {
      // 锁定状态,继续搜索是否存在该类的实例 //
      lock (_syslock)
      {
      // 如果不存在,在锁定状态下实例化出一个实例 //
      if (_instance == null)
      {
      _instance = new Singleton();
      return _instance;
      }
      else // 锁定状态下,存在该类实例,直接返回 //
      {
      return _instance;
      }
      }
      }
      // 该实例本身就已经存在了,直接返回 //
      return _instance;
      }
      }
  4. 泛型单例模式

    4.1 在一个案例中,我们可能需要使用到不止一个单例模式类,甚至更多。那么此时,使用泛型单例模式模板来实现单例模式,我们可以有两种不同的方法来实现它:

    4.2首先我们来看下泛型模板,我们对泛型类进行约束,T只能是一个Class,并且有一个公共无参构造函数,代码如下:

    using System;
    using UnityEngine; public class SingletonProvider<T> where T : class ,new()
    {
    private SingletonProvider()
    {
    } private static T _instance;
    // 用于lock块的对象
    private static readonly object _synclock = new object(); public static T Instance
    {
    get
    {
    if (_instance == null)
    {
    lock (_synclock)
    {
    if (_instance == null)
    {
    // 若T class具有私有构造函数,那么则无法使用SingletonProvider<T>来实例化new T();
    _instance = new T();
    //测试用,如果T类型创建了实例,则输出它的类型名称
    Debug.Log("{0}:创建了单例对象" + typeof(T).Name);
    }
    }
    }
    return _instance;
    }
    set { _instance = value; }
    }
    }
    • 4.2.1 然后我们定义了一个网络连接类 NetIO,使用单例提供类中的泛型T替代为具体的网络连接类进行使用:
    • 4.2.2 使用具体类替代泛型,用泛型单例提供类对该具体类达到提供唯一实例的单例实现效果:
    • 4.2.3 具体类中定义了字段NetIoCreateTime来存储该类实例化的时间,进行下一步分析该类实例是否是唯一实例,具体类代码如下:
      public class NetIO
      { public static NetIO GetInstance()
      {
      return SingletonProvider<NetIO>.Instance;
      } public NetIO()
      {
      this.NetIoCreateTime = DateTime.Now;
      } public DateTime NetIoCreateTime
      {
      get { return _ct; }
      set { _ct = value; }
      } private DateTime _ct;
      }
    • 4.2.4 在Unity中Update参数中调用该类,对该类创建时间进行输出
      public void Update()
      {
      Debug.Log(NetIO.GetInstance().NetIoCreateTime);
      }
    • 4.2.5 测试结果如下:
    • 4.2.6 所有创建时间都一致,证明该类提供单例提供类中的泛型替代,达到了单例模式的效果,提供了该类的唯一实例访问

    4.3 在一个案例中,我们可能需要使用到不止一个单例模式类,甚至更多。那么此时,使用泛型单例模式模板来实现单例模式,我们可以有两种不同的方法来实现它:

    • 4.3.1 首先我们来看下泛型模板,我们对泛型类进行约束,T只能是一个Class,并且有一个公共无参构造函数,代码如下:

      using System;
      using UnityEngine; public class SingletonProvider<T> where T : class ,new()
      {
      private SingletonProvider()
      {
      } private static T _instance;
      // 用于lock块的对象
      private static readonly object _synclock = new object(); public static T Instance
      {
      get
      {
      if (_instance == null)
      {
      lock (_synclock)
      {
      if (_instance == null)
      {
      // 若T class具有私有构造函数,那么则无法使用SingletonProvider<T>来实例化new T();
      _instance = new T();
      //测试用,如果T类型创建了实例,则输出它的类型名称
      Debug.Log("{0}:创建了单例对象" + typeof(T).Name);
      }
      }
      }
      return _instance;
      }
      set { _instance = value; }
      }
      }
    • 4.3.2 然后我们定义了一个网络连接类 NetIO,使用单例提供类中的泛型T替代为具体的网络连接类进行使用:
      1. 使用具体类替代泛型,用泛型单例提供类对该具体类达到提供唯一实例的单例实现效果:
      2. 具体类中定义了字段NetIoCreateTime来存储该类实例化的时间,进行下一步分析该类实例是否是唯一实例,具体类代码如下:
        public class NetIO
        { public static NetIO GetInstance()
        {
        return SingletonProvider<NetIO>.Instance;
        } public NetIO()
        {
        this.NetIoCreateTime = DateTime.Now;
        } public DateTime NetIoCreateTime
        {
        get { return _ct; }
        set { _ct = value; }
        } private DateTime _ct;
        }
      3. 在Unity中Update参数中调用该类,对该类创建时间进行输出
        public void Update()
        {
        Debug.Log(NetIO.GetInstance().NetIoCreateTime);
        }
      4. 测试结果如下:
      5. 所有创建时间都一致,证明该类提供单例提供类中的泛型替代,达到了单例模式的效果,提供了该类的唯一实例访问

Unity中C#单例模式使用总结的更多相关文章

  1. MonoSingleton——Unity中的单例模式

    Unity中有很多特别的类需要以单例模式呈现,比如全局的UI管理类,各种缓存池,以及新手导航类等等.而Unity中,因为所有继承自Monobehaviour的脚本在实现的时候都是单线程的,所以像网上流 ...

  2. 【Unity3D基础教程】给初学者看的Unity教程(七):在Unity中构建健壮的单例模式(Singleton)

    作者:王选易,出处:http://www.cnblogs.com/neverdie/ 欢迎转载,也请保留这段声明.如果你喜欢这篇文章,请点推荐.谢谢! 该博客中的代码均出自我的开源项目 : 迷你微信 ...

  3. Unity中的万能对象池

    本文为博主原创文章,欢迎转载.请保留博主链接http://blog.csdn.net/andrewfan Unity编程标准导引-3.4 Unity中的万能对象池 本节通过一个简单的射击子弹的示例来介 ...

  4. 【《Effective C#》提炼总结】提高Unity中C#代码质量的21条准则

    作者:Williammao, 腾讯移动客户端开发工程师 商业转载请联系腾讯WeTest获得授权,非商业转载请注明出处. 原文链接:http://wetest.qq.com/lab/view/290.h ...

  5. 【《Effective C#》提炼总结】提高Unity中C#代码质量的22条准则

    引言 原则1尽可能地使用属性而不是可直接访问的数据成员 原则2偏向于使用运行时常量而不是编译时常量 原则3 推荐使用is 或as操作符而不是强制类型转换 原则4 推荐使用条件属性而不是if条件编译 原 ...

  6. Unity中利用委托与监听解耦合的思路

    这篇随笔是一篇记录性的随笔,记录了从http://www.sikiedu.com/my/course/304,这门课程中学到的内容,附带了一些自己的思考. 一.单例模式的应用 首先假想一种情况,现在需 ...

  7. 【转】Effective C#观后感之提高Unity中C#代码质量的21条准则

    转自:http://blog.csdn.net/swj524152416/article/details/75418162 我们知道,在C++领域,作为进阶阅读材料,必看的书是<Effectiv ...

  8. Unity编程标准导引-3.4 Unity中的对象池

    本文为博主原创文章,欢迎转载.请保留博主链接http://blog.csdn.net/andrewfan Unity编程标准导引-3.4 Unity中的对象池 本节通过一个简单的射击子弹的示例来介绍T ...

  9. Unity中使用Attribute

    Attribute是c#的语言特性 msdn说明如下: The Attribute class associates predefined system information or user-def ...

随机推荐

  1. HDU 1050(楼道搬桌子问题)(不是贪心解法,思路很新颖)

    Moving Tables Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tot ...

  2. 一篇很好的解释了.Net Core, .Net Framework, .Net standard library, Xamarin 之间关系的文章 (转载)

    Introducing .NET Standard In my last post, I talked about how we want to make porting to .NET Core e ...

  3. 我告诉你 ,一个 window免费系统下载的网站!

    一个 window免费系统下载的网站! https://msdn.itellyou.cn/

  4. Java单例模式几种实现方式

    在平时的工作.学员的学习以及面试过程中,单例模式作为一种常用的设计模式,会经常被面试官问到,甚至笔试会要求学员现场默写,下面将会就单例模式的实现思路和几种常见的实现方式进行简单的分享. 单例模式,是一 ...

  5. Node.js——fs模块(文件系统),创建、删除目录(文件),读取写入文件流

    /* 1. fs.stat 检测是文件还是目录(目录 文件是否存在) 2. fs.mkdir 创建目录 (创建之前先判断是否存在) 3. fs.writeFile 写入文件(文件不存在就创建,但不能创 ...

  6. windbg调试虚拟机XP系统

    一.先介绍一下被调试的虚拟机系统环境: 虚拟机:vmware workstation 10.0版本 虚拟机操作系统: Microsoft windows xp professional 2002 se ...

  7. 网易云易盾CTO朱浩齐:我们是如何用AI赋能内容安全?

    本文由  网易云发布. 5月19日,LiveVideoStack携手网易云易盾,共同打造了“娱乐多媒体开发应用实践”专题,帮助开发者和泛娱乐平台运营人员,提升技术能力,突破难点,拓展思路与视野. 在专 ...

  8. Oracle入门第六天(下)——高级子查询

    一.概述 主要内容: 二.子查询介绍 1.简单子查询(WHERE子查询) SELECT last_name FROM employees WHERE salary > (SELECT salar ...

  9. Flex copy and paste

    <?xml version="1.0" encoding="utf-8"?> <mx:WindowedApplication xmlns:mx ...

  10. 20155307 2016-2017-2 《Java程序设计》第4周学习总结

    20155307 2016-2017-2 <Java程序设计>第4周学习总结 教材学习内容总结 所谓继承,讲的就是出现很多很多相同的部分的话,就把这个部分变成"父类", ...