当前位置: 首页 > article >正文

2. 继承Mono的单例模式基类

前提

  1. 继承MonoBehaviour的脚本不能new
  2. 继承MonoBehaviour的脚本一定得依附在GameObject上

实现挂载式的单例模式基类

挂载式 继承Mono的单例模式基类


/// <summary>
/// 挂载式 继承Mono的单例模式基类
/// </summary>
/// <typeparam name="T"></typeparam>
public class SingletonMono<T>: MonoBehaviour where T:MonoBehaviour
{
    private static T instance;

    public static T Instance
    {
        get
        {
            return instance;
        }
    }


    // 子类可能要重写Awake函数,所以设置为虚函数
    protected virtual void Awake()
    {
        
        instance = this as T;
        
    }
}

TestMgr2继承单例模式的基类,然后把脚本挂在到游戏对象上去,才能在Awake赋值


public class TestMgr2 : SingletonMono<TestMgr2>
{
    private int i;

    protected override void Awake()
    {
        // 重写Awake时候,这里不能省略
        base.Awake();

        i = 0;
    }
    public void TestLog()
    {
        print("TestMgr2" + i);
    }
    
}

测试

public class Main : MonoBehaviour
{

    private void Start()
    {
        TestMgr2.Instance.TestLog();
    }

}

缺点

因为很容易被破坏单例模式的唯一性
1.挂载多个脚本
2.切换场景回来时,由于场景放置了挂载脚本的对象,回到该场景时 又会有一个该单例模式对象
3.还可以通过代码动态的添加多个该脚本 也会破坏唯一性

修改Awake


/// <summary>
/// 挂载式 继承Mono的单例模式基类
/// </summary>
/// <typeparam name="T"></typeparam>
public class SingletonMono<T>: MonoBehaviour where T:MonoBehaviour
{
    private static T instance;

    public static T Instance
    {
        get
        {
            return instance;
        }
    }


    // 子类可能要重写Awake函数,所以设置为虚函数
    protected virtual void Awake()
    {
        //已经存在一个对应的单例模式对象了 不需要在有一个了
        if(instance != null)
        {
        	// 只会移除脚本
            Destroy(this);
            return;
        }
        instance = this as T;
        //我们挂载继承该单例模式基类的脚本后 依附的对象过场景时就不会被移除了
        //就可以保证在游戏的整个生命周期中都存在 
        DontDestroyOnLoad(this.gameObject);
    }
}

实现自动挂载式的单例模式基类

创建单例模式基类SingletonAutoMono,实现动态挂载脚本


/// <summary>
/// 自动挂载式的 继承Mono的单例模式基类
/// 推荐使用 
/// 无需手动挂载 无需动态添加 无需关心切场景带来的问题
/// </summary>
/// <typeparam name="T"></typeparam>
public class SingletonAutoMono<T> : MonoBehaviour where T:MonoBehaviour
{
    private static T instance;

    public static T Instance
    {
        get
        {
            if(instance == null)
            {
                //动态创建 动态挂载
                //在场景上创建空物体
                GameObject obj = new GameObject();
                //得到T脚本的类名 为对象改名 这样再编辑器中可以明确的看到该
                //单例模式脚本对象依附的GameObject
                obj.name = typeof(T).ToString();
                //动态挂载对应的 单例模式脚本
                instance = obj.AddComponent<T>();
                //过场景时不移除对象 保证它在整个游戏生命周期中都存在
                DontDestroyOnLoad(obj);
            }
            return instance;
        }
    }

}

Test2Mgr2继承单例模式基类

public class Test2Mgr2 : SingletonAutoMono<Test2Mgr2>
{
    public void TestLog()
    {
        print("Test2Mgr2");
    }
}

测试

public class Main : MonoBehaviour
{

    private void Start()
    {
        Test2Mgr2.Instance.TestLog();
    }

}

潜在的安全问题

  1. 构造函数问题:
    继承MonoBehaviour的函数,不能new,所以不用担心公共构造函数
  2. 多线程问题:
    Unity主线程中相关内容,不允许其他线程直接调用,很少有这样的需求,所以也不用太担心
  3. 重复挂载问题:
    1. 手动重复挂载
    2. 代码重复添加

需要人为干涉,定规则,或者通过代码逻辑强制处理


http://www.kler.cn/a/355079.html

相关文章:

  • CI/CD是什么?
  • InnoDB引擎的内存结构
  • 基于ceres优化的3d激光雷达开源算法
  • 内部知识库的未来展望:技术融合与用户体验的双重升级
  • Zerotier + VSCode远程连接实验室的服务器、Xshell连接远程服务器
  • 后端-redis
  • 一文搞懂模型倍率怎么计算的,以及模型分组倍率原理!
  • Java | Leetcode Java题解之第480题滑动窗口中位数
  • 决策树C4.5算法详解及实现
  • openEuler-22.03-SP4离线编译安装ZLMediaKit
  • A Survey on 3D Gaussian Splatting 整理
  • XML 和 SimpleXML 简介
  • linux环境下的程序设计与git操作
  • 【MySQL】入门篇—基本数据类型:NULL值的概念
  • 利用mydumper从MySQL迁移数据到OceanBase数据库命令记录
  • PHP学习记录-编辑器推荐和本地环境的安装
  • 锁定云轴科技ZStack主题演讲,10月19日中国云计算基础架构开发者大会见
  • WHAT - Antd 定制主题之预设算法
  • 【论文笔记】Fine-tuned CLIP Models are Efficient Video Learners
  • 代替AD作为身份认证组件,深信服零信任aTrust与宁盾身份目录实现互操作
  • leetcode 刷题day43动态规划Part12(115.不同的子序列、583. 两个字符串的删除操作、72. 编辑距离)
  • 一键获取每日股票数据,自动更新,尽在掌握
  • Autosar软件组件概述
  • 数字图像处理:图像复原应用
  • java 调用 k8s 的 apiserver
  • 公开选拔!产业实践教授