Unity中C#单例模式使用总结
一、单例模式优点
- 单例模式核心在于对于某个单例类,在系统中同时只存在唯一一个实例,并且该实例容易被外界所访问;
- 意味着在内存中,只存在一个实例,减少了内存开销;
二、单例模式特点
- 只存在唯一一个实例;
- 提供统一对外访问接口,使得全局可对该单例的唯一实例进行访问;
- 自行实例化(私有构造函数,不允许外界对其进行实例化)。
三、单例模式使用
- 资源管理器,资源对象数据的加载和卸载(无状态不需要实例化的对象);
- 单一客户端连接服务器等;
- 生命周期在游戏中永不消毁的对象。
四、单例模式注意点
- 注意线程安全问题,在多线程、高并发的情况下,可能同时产生多个实例,违背了单例模式。
- Unity中如果过度使用单例模式,将会导致代码耦合度非常高,脚本与脚本之间的耦合,代码的后续拓展变得非常麻烦。一个过分依赖单例模式的开发者不能成为一个好的开发者,也不会去接触到更多优秀的设计模式。个人推荐ECS 实体 - 组件式编程。
- Unity中暂时不需要考虑多线程问题,Unity就只有一个主线程和开启多个辅助协程,不会出现多线程并发问题。
- 控制游戏对象的生成和销毁并不建议使用单例模式,可通过主游戏逻辑InGame进行事件下发,自行管理Update,使用工厂来进行对象的创建和销毁。
五、单例模式常见模式
懒汉模式(最常用)
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.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.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.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替代为具体的网络连接类进行使用:
- 使用具体类替代泛型,用泛型单例提供类对该具体类达到提供唯一实例的单例实现效果:
- 具体类中定义了字段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;
} - 在Unity中Update参数中调用该类,对该类创建时间进行输出
public void Update()
{
Debug.Log(NetIO.GetInstance().NetIoCreateTime);
} - 测试结果如下:

- 所有创建时间都一致,证明该类提供单例提供类中的泛型替代,达到了单例模式的效果,提供了该类的唯一实例访问
- 4.3.1 首先我们来看下泛型模板,我们对泛型类进行约束,T只能是一个Class,并且有一个公共无参构造函数,代码如下:
Unity中C#单例模式使用总结的更多相关文章
- MonoSingleton——Unity中的单例模式
Unity中有很多特别的类需要以单例模式呈现,比如全局的UI管理类,各种缓存池,以及新手导航类等等.而Unity中,因为所有继承自Monobehaviour的脚本在实现的时候都是单线程的,所以像网上流 ...
- 【Unity3D基础教程】给初学者看的Unity教程(七):在Unity中构建健壮的单例模式(Singleton)
作者:王选易,出处:http://www.cnblogs.com/neverdie/ 欢迎转载,也请保留这段声明.如果你喜欢这篇文章,请点推荐.谢谢! 该博客中的代码均出自我的开源项目 : 迷你微信 ...
- Unity中的万能对象池
本文为博主原创文章,欢迎转载.请保留博主链接http://blog.csdn.net/andrewfan Unity编程标准导引-3.4 Unity中的万能对象池 本节通过一个简单的射击子弹的示例来介 ...
- 【《Effective C#》提炼总结】提高Unity中C#代码质量的21条准则
作者:Williammao, 腾讯移动客户端开发工程师 商业转载请联系腾讯WeTest获得授权,非商业转载请注明出处. 原文链接:http://wetest.qq.com/lab/view/290.h ...
- 【《Effective C#》提炼总结】提高Unity中C#代码质量的22条准则
引言 原则1尽可能地使用属性而不是可直接访问的数据成员 原则2偏向于使用运行时常量而不是编译时常量 原则3 推荐使用is 或as操作符而不是强制类型转换 原则4 推荐使用条件属性而不是if条件编译 原 ...
- Unity中利用委托与监听解耦合的思路
这篇随笔是一篇记录性的随笔,记录了从http://www.sikiedu.com/my/course/304,这门课程中学到的内容,附带了一些自己的思考. 一.单例模式的应用 首先假想一种情况,现在需 ...
- 【转】Effective C#观后感之提高Unity中C#代码质量的21条准则
转自:http://blog.csdn.net/swj524152416/article/details/75418162 我们知道,在C++领域,作为进阶阅读材料,必看的书是<Effectiv ...
- Unity编程标准导引-3.4 Unity中的对象池
本文为博主原创文章,欢迎转载.请保留博主链接http://blog.csdn.net/andrewfan Unity编程标准导引-3.4 Unity中的对象池 本节通过一个简单的射击子弹的示例来介绍T ...
- Unity中使用Attribute
Attribute是c#的语言特性 msdn说明如下: The Attribute class associates predefined system information or user-def ...
随机推荐
- consul搭建(初步)
http://www.cnblogs.com/java-zhao/p/5375132.html https://blog.csdn.net/u010246789/article/details/517 ...
- 404 Note Found 队-Alpha1
目录 组员情况 组员1(组长):胡绪佩 组员2:胡青元 组员3:庄卉 组员4:家灿 组员5:凯琳 组员6:丹丹 组员7:家伟 组员8:政演 组员9:黄鸿杰 组员10:刘一好 组员11:何宇恒 展示组内 ...
- 阅读Deep Packet Inspection based Application-Aware Traffic Control for Software Defined Networks
Deep Packet Inspection based Application-Aware Traffic Control for Software Defined Networks Globlec ...
- 3.Spring Cloud初相识--------Ribbon客户端负载均衡
前言: 在生产环境中,未避免单点故障,每个微服务都会做高可用部署. 通白的说,就是每一个一模一样的服务会根据需求提供多分在多台机器上. 那么在大并发的情况下,如何分配服务可以快速得到响应,就成为了我们 ...
- iOS 推送功能打包后获取不到deviceToken
公司项目用ionic3构建, 用了极光推送插件(cordova-plugin-jpush). 开发时一切将各种Bundle Id, 推送证书等都绑定完测试一切正常. 可是要给测试人员打Ad-Hoc包时 ...
- 1001. 温度转换 (Standard IO)
1001. 温度转换 (Standard IO) 时间限制: 1000 ms 空间限制: 262144 KB 具体限制 题目描述 将输入的华氏温度转换为摄氏温度.由华氏温度F与摄氏温度C的转换 ...
- java Clob类型 转String
1.我的数据库是oracle11g 遇到取出来的字段是clob类型,但是所需要的是string类型,写一个转换函数就可以解决问题了. // Clob类型 转String public String C ...
- ES6学习之-let 和const命令
let 和const命令 let命令 用来声明变量,类似于var .let声明的变量 只在let命令所在的代码块内有效. 在for循环里也是如此 每次循环其实都是一个代码块 function fn() ...
- Java实例 Part6:Java中的克隆
目录 Part6:Java中的克隆 Example01:Java对象的假克隆 Example02:Java对象的浅克隆 Example03:Java对象的深克隆 Example04:序列化与对象克隆 ...
- Hadoop(二)CentOS7.5搭建Hadoop2.7.6完全分布式集群
一 完全分布式集群(单点) Hadoop官方地址:http://hadoop.apache.org/ 1 准备3台客户机 1.1防火墙,静态IP,主机名 关闭防火墙,设置静态IP,主机名此处略,参考 ...