[Unity] 单例基类的实现方式
Unity单例基类的实现方式
游戏开发的过程中我们经常会将各种“Manger”类设为单例,以实现单一资源的全局访问。
Unity中的单例一般分为两类,一种是直接继承自Object的普通单例,还有一种是需要继承MonoBehaviour的Mono单例。接下来我将会讲解这两种单例基类的实现方式。
注意:由于Unity的限制,项目中一般不会出现多线程对单例竞争访问的情况,因此以下实现都是以单线程为前提考虑的,但你依旧可以在此基础上使用经典的双重检查实现线程安全的初始化。
普通单例基类
为了使基类足够通用,我们需要用泛型T
来表示单例的类型。并且这个单例也必须是个引用类型(class)(你也不会想用int或struct作为单例对象吧),因此要加上where T : class
约束。
public abstract class Singleton<T> where T : class
{
private static T _instance;
public static T Instance
{
get
{
// ...
}
}
}
接下来我们就要处理单例的初始化,一般而言你可能会直接使用_instance = new T()
进行初始化,但这会带来一个问题——子类T
必须有public的无参构造函数。这就表示编译器无法阻止你或别人在程序的其地方使用new T()
,导致单例的唯一性被破坏,作为一个强迫症并不能容忍这种潜在的风险。
public abstract class Singleton<T> where T : class, new() // 注意这里要添加new()约束
{
private static T _instance;
public static T Instance
{
get
{
if (_instance == null)
{
_instance = new T();
}
return _instance;
}
}
}
解决方案也很简单,我们先将T
的构造函数设为private
,防止被外部调用。
在我们需要初始化单例的时候,就通过反射来获取对象T
的私有无参构造函数。
注意,这里强调了是私有的无参构造函数,因为我们在GetConstructor
的参数中使用了BindingFlags.NonPublic
,因此只能获取私有的构造函数。这也是对子类的约束,使其必须将无参构造函数设为私有,否则就会抛出异常。
public abstract class Singleton<T> where T : class
{
private static T _instance;
public static T Instance
{
get
{
if (_instance == null)
{
// 反射获取私有无参构造函数
ConstructorInfo ctor = typeof(T).GetConstructor(BindingFlags.Instance | BindingFlags.NonPublic, null, Type.EmptyTypes, null);
if (ctor == null)
{
throw new InvalidOperationException("Singleton classes must have exactly one private constructor.");
}
// 初始化
_instance = ctor.Invoke(null) as T;
}
return _instance;
}
}
}
Mono单例基类
Mono单例的实例化方式比较特殊,首先MonoBehaviour
是不能被new
的,所以不需要考虑构造函数的问题。
想要实例化一个MonoBehaviour
对象,就需要将其挂载到场景的某个GameObject
上。因此我们可以通过以下代码,创建一个GameObject对象,然后进行将T
挂载到该对象上,以获取T
的实例。需要注意的是,Unity在切换场景时会销毁所有对象,挂载在对象上的组件也会一并消失,因此我们需要使用DontDestroyOnLoad
标记对象,以防止被销毁。
GameObject singletonObject = new GameObject(typeof(T).Name + "(Singleton)");
DontDestroyOnLoad(singletonObject); // 保留在场景切换时不被销毁
_instance = singletonObject.AddComponent<T>();
最终代码如下:
public abstract class SingletonMono<T> : MonoBehaviour where T : MonoBehaviour
{
private static T _instance;
public static T Instance
{
get
{
if (_instance == null)
{
// 在场景中查找是否已存在该类型的实例
_instance = FindObjectOfType<T>();
// 如果场景中不存在该类型的实例,则创建一个新的GameObject并添加该组件
if (_instance == null)
{
GameObject singletonObject = new GameObject(typeof(T).Name + "(Singleton)");
DontDestroyOnLoad(singletonObject); // 保留在场景切换时不被销毁
_instance = singletonObject.AddComponent<T>();
}
}
return _instance;
}
}
}
本文发布于2024年5月22日
最后修改于2024年5月22日
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!