Unity中有很多特别的类需要以单例模式呈现,比如全局的UI管理类,各种缓存池,以及新手导航类等等。而Unity中,因为所有继承自Monobehaviour的脚本在实现的时候都是单线程的,所以像网上流传的一些C#的实现方式就显得不那么的实用了。

很多国内的公司所使用的MonoSingleton都是有问题的,比如像Easytouch中关于单例是这样实现中有这样一段代码。

        public static T instance
{
get
{
if (m_Instance == null)
{
m_Instance = GameObject.FindObjectOfType(typeof(T)) as T;//1这里耗费性能,有风险
if (m_Instance == null)//2
{
m_Instance = new GameObject("Singleton of " + typeof(T).ToString(), typeof(T)).GetComponent<T>();
m_Instance.Init();
}
}
return m_Instance;
}
}

  

那么我标注的两处就是代码当中不正确的地方。2处这是明显的套用了多线程的单例实现方式,而实际上,在单线程模式当中这个判断并没有意义。而1中,直接对全场景进行搜索的过程其本身就很浪费性能。那么正确的实现方式是什么呢?

首先,我们需要一个全局变量,比如,先建立一个全局类Global

public abstract class Global : MonoBehaviour
{
public static HashSet<string> Singleton=new HashSet<string>();
}

每次建立都将类名存进HashSet当中,那么上面那段代码就可以改成这样。

    public static T instance
{
get
{
if (m_Instance == null)
{
var name = "Singleton of " + typeof(T).ToString();
var flag = Global.Singleton.Contains(name);
if (!flag)
{
m_Instance = new GameObject(name, typeof(T)).GetComponent<T>();
m_Instance.Init();
Global.Singleton.Add(name);
}
}
return m_Instance;
}
}

可能您要说了,我已经有了一个全局类了,那么难道还要再填一个东西?我只想直接用,用没有更简便的方法。您要说更好,不一定,但是更简便,确实有的。我们这里可以用上互斥类Mutex的类,那么上面那段代码就可以改成下面这样:

    public static T instance
{
get
{
if (m_Instance == null)//注意,此处在实际中只执行一次。
{
bool createdNew;
var name = "Singleton of " + typeof(T).ToString();
Mutex mutex = new Mutex(false, name, out createdNew);
if (createdNew)
{
m_Instance = new GameObject(name, typeof(T)).GetComponent<T>();
m_Instance.Init();
}
}
return m_Instance;
}
}

但这只是说,我如果在其他地方操作这个单例,而这个单例还必须新建一个游戏物体,还必须挂在上面,挂在游戏物体上就至少要有一个transform组件。那么我可不可以直接挂在物体上,那该怎么办?如果我挂多了该怎么办?

有办法,这里我们利用Awake()方法

    private void Awake()
{
if (m_Instance == null)
{
bool createdNew;
var name = "Singleton of " + typeof(T).ToString();
Mutex mutex = new Mutex(false, name, out createdNew);
if (createdNew)
{
m_Instance = this as T;
m_Instance.Init();
}
}
else
{
Destroy(this);
}
}

这样就可以保证运行时的单例了。那么完整的MonoSingleton还需要一些细节。比如在我的单例基类中,我设计成抽象类,提供了两个抽象函数,分别是初始化和逆初始化。之所以这么做就是为了在我们继承MonoSingleton的时候想一想,是不是必须要把这个类做成单例的,它一定是有单例的必要所以才是单例的,而不是将单例当静态使用。

using UnityEngine;
using System.Collections;
using System.Threading;
/// <summary>
/// 单例基类,提供两个抽象函数Init 和 DisInit 初始化和逆初始化过程。
/// </summary>
/// <typeparam name="T"></typeparam>
public abstract class MonoSingleton<T> : MonoBehaviour
where T : MonoSingleton<T>
{ private static T m_Instance = null;
private static string name;
private static Mutex mutex;
public static T instance
{
get
{
if (m_Instance == null)
{
if ( IsSingle())
{
m_Instance = new GameObject(name, typeof(T)).GetComponent<T>();
m_Instance.Init();
}
}
return m_Instance;
}
} private static bool IsSingle()
{
bool createdNew;
name = "Singleton of " + typeof(T).ToString();
mutex = new Mutex(false, name, out createdNew);
return createdNew;
} private void Awake()
{
if (m_Instance == null)
{
if (IsSingle())
{
m_Instance = this as T;
m_Instance.Init();
}
}
else
{
Destroy(this);
}
} protected abstract void Init();
protected abstract void DisInit();
private void OnDestory()
{
if (m_Instance!=null)
{
mutex.ReleaseMutex();
DisInit();
m_Instance = null;
}
}
private void OnApplicationQuit()
{
mutex.ReleaseMutex();
}
}

  

  

  

  

  

最新文章

  1. C# 自定义Section
  2. C#防止反编译
  3. command line
  4. [WP8.1UI控件编程]Windows Phone VirtualizingStackPanel、ItemsStackPanel和ItemsWrapGrid虚拟化排列布局控件
  5. Dubbo架构设计详解(转自shiyanjun.cn)
  6. ucenter 通信成功后 apps.php无误后 该做的事
  7. SQL Server面试题
  8. java运行内存分配图(转)
  9. linux内核内存管理(zone_dma zone_normal zone_highmem)
  10. 在MyEclipse 2014中给Spket增加ExtJS提示
  11. Java calendar获取月份注意事项
  12. Ubuntu16.04+cuda9.0+matlab+opencv3.3+caffe服务器配置(问题汇总)
  13. Xcode中编译iOS程序,运行出错:Thread 1: signal SIGABRT
  14. python 实现网页 自动登录
  15. Django框架(三)
  16. sql server 备份与恢复系列七 页面还原
  17. 【刷题】BZOJ 4254 Aerial Tramway
  18. PHP 中的对象传递
  19. python调用函数
  20. react-native init安装指定版本的react-native

热门文章

  1. python的多线程threading
  2. keycloak学习
  3. 【Python初级】由判定回文数想到的,关于深浅复制,以及字符串反转的问题
  4. 【UOJ 179】 #179. 线性规划 (单纯形法)
  5. [BZOJ4552][TJOI2016&amp;&amp;HEOI2016]排序(二分答案+线段树/线段树分裂与合并)
  6. 【动态规划/多重背包问题】POJ1014-Dividing
  7. VIM选项配置说明
  8. LCA POJ 1330 Nearest Common Ancestors
  9. python学习一月总结_汇总大牛们的思想_值得收藏
  10. 卡尔曼滤波(Kalman Filter)在目标边框预测中的应用