首页 > 其他分享 >[Unity] 单例基类的实现方式

[Unity] 单例基类的实现方式

时间:2024-05-22 15:10:07浏览次数:13  
标签:class instance Unity 单例 基类 new public 构造函数

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日

标签:class,instance,Unity,单例,基类,new,public,构造函数
From: https://www.cnblogs.com/ThousandPine/p/18206244

相关文章

  • Unity安卓IOS一键打包
    添加菜单构建按钮,使用下面API进行构建,注意设置和配置等usingSystem;usingSystem.IO;usingAssetBundles;usingLiXiaoQian.Common.Editor.Tools;usingUnityEditor;usingUnityEngine;///打包工具publicclassBuildTool{[MenuItem("Tools/构建/Android平台")]......
  • Unity控制物体有放缩弹一弹的效果
     usingUnityEngine;usingSystem.Collections;//控制有放缩弹一弹的效果publicclassScaleController:MonoBehaviour{boolismax=false;publicfloatscaleSpeed=0.003f;//可调速度publicfloatscalemax=2,scalemin=1.5f;//可调范......
  • 【unity】在EditorWindow添加自定义的Toolbar按钮
    好久没写了,最近做工具,写了个EditorWindow,为了让使用者方便查看这个工具的文档,想着在导航栏加个问号按钮,点一下打开说明文档就完事~查了下unity手册,发现Unity提供了一个ShowButton 方法,用于在自定义Editor窗口的工具栏中添加自定义内容,下面是实现的例子:privateGUIStyle......
  • Unity制作一个协程管理工具
    IEnumeratorToolusingSystem;usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;publicclassIEnumeratorTool:MonoBehaviour{///<summary>///压入的action任务///</summary>publicclassActionTask......
  • Unity设置UI和Render的渲染层级
    通过给UI或物体挂载下面脚本,来设置层级usingUnityEngine;usingSystem.Collections;usingUnityEngine.UI;namespaceCommon{//设置UI和render的层级publicclassUIDepth:MonoBehaviour{publicintorder;publicboolisUI=true;......
  • Unity性能优化:什么是内存泄露?
    内存泄漏是优化方面的名词,主要是由于不再使用的资源没有及时清理,来释放内存,造成内存的浪费,造成系统卡顿。 或者说,内存就像花呗,额度就这么多,有借要有还,而且手里有闲钱的时候就记得还,以保证内存的充足,如果占着不用,就会在其他需要使用的时候内存不足,就容易崩溃出现问题。 Unity......
  • Unity制作一个定时器Timer
    Timer和TimerManager代码usingSystem.Collections;usingUnityEngine;publicclassTimer:MonoBehaviour{publicdelegatevoidNotifier();publicNotifieronTimer;publicNotifieronTimerReset;publicNotifieronTimerComplete;publicfl......
  • Unity制作一个BroadcastUI 跑马灯文字广播
     usingDG.Tweening;usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;usingUnityEngine.UI;usingUtils;//挂在UI上面publicclassBroadcastUI:MonoBehaviour{privateboolinited=false;privateBroadcastManbm;......
  • Unity查找物体和组件的方法
    一、找物体:①GameObject:a).Find(stringname)通过物体的名字查找b).FindWithTag(stringtag);通过标签获取添加该标签的一个物体c).FindObjectOfType();依据组件类型d).FindGameObjectsWithTag(stringtag)通过标签获取所有添加该标签的物体数组返回一个组合 ②Transform......
  • Unity的UnityEngine.EventSystems中的接口
    一、IPointerDownHandler,IPointerUpHandler,IPointerClickHandler,IPointerEnterHandler,IPointerExitHandlerpublicvoidOnPointerClick(PointerEventDataeventData){Debug.Log("OnPointerClick,鼠标点击,在点击之后抬起时响应");}publicvoidOnP......