Unity 游戏框架搭建 2019 (四十八) MonoBehaviourSimplify 中的消息策略完善

本文主要是介绍Unity 游戏框架搭建 2019 (四十八) MonoBehaviourSimplify 中的消息策略完善,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上一篇,笔者说,MonoBehaviourSimplify 中的消息策略还有一些小问题。我们在这篇试着解决一下。

先贴出来代码:

using System;
using System.Collections.Generic;namespace QFramework
{public abstract partial class MonoBehaviourSimplify{Dictionary<string, Action<object>> mMsgRegisterRecorder = new Dictionary<string, Action<object>>();protected void RegisterMsg(string msgName, Action<object> onMsgReceived){MsgDispatcher.Register(msgName, onMsgReceived);mMsgRegisterRecorder.Add(msgName, onMsgReceived);}private void OnDestroy(){OnBeforeDestroy();foreach (var keyValuePair in mMsgRegisterRecorder){MsgDispatcher.UnRegister(keyValuePair.Key,keyValuePair.Value);                }mMsgRegisterRecorder.Clear();}protected abstract void OnBeforeDestroy();}public class B : MonoBehaviourSimplify{private void Awake(){RegisterMsg("Do", DoSomething);RegisterMsg("DO1", _ => { });RegisterMsg("DO2", _ => { });RegisterMsg("DO3", _ => { });}void DoSomething(object data){// do something}protected override void OnBeforeDestroy(){}}
}

我们是使用字典进行注册消息的记录的,使用字典就要保证字典中的 key 是唯一的。而我们很可能在一个脚本中对一个关键字注册多次,这样用字典这个数据结构就显得不合理了。

相比字典,List 更合适,因为我们有有可能有重复的内容,而字典更适合做一些查询工作,但是 List 并不支持键值对,怎么办呢?

我们只好创建一个结构来存储我们的消息名和对应的委托,这个结构是一个类叫做 MsgRecord

消息策略部分的代码如下:

    public abstract partial class MonoBehaviourSimplify{List<MsgRecord> mMsgRecorder = new List<MsgRecord>();private class MsgRecord{public string Name;public Action<object> OnMsgReceived;}protected void RegisterMsg(string msgName, Action<object> onMsgReceived){MsgDispatcher.Register(msgName, onMsgReceived);mMsgRecorder.Add(new MsgRecord{Name = msgName,OnMsgReceived = onMsgReceived});}private void OnDestroy(){OnBeforeDestroy();foreach (var msgRecord in mMsgRecorder){MsgDispatcher.UnRegister(msgRecord.Name,msgRecord.OnMsgReceived);                }mMsgRecorder.Clear();}protected abstract void OnBeforeDestroy();}

代码比较简单。

而我们的示例代码,如下,增加了一行重复注册的代码。

    public class B : MonoBehaviourSimplify{private void Awake(){RegisterMsg("Do", DoSomething);RegisterMsg("Do", DoSomething);RegisterMsg("DO1", _ => { });RegisterMsg("DO2", _ => { });RegisterMsg("DO3", _ => { });}void DoSomething(object data){// do something}protected override void OnBeforeDestroy(){}}

而我们的 MonoBehaviourSimplify 内部实现发生了天翻地覆的变化,也没有对我们的示例代码产生一点影响,这叫封装。

那么到这里,我们的消息策略还有问题吗?

还有的,问题在创建 MsgRecord 的部分。
如下:

mMsgRecorder.Add(new MsgRecord
{Name = msgName,OnMsgReceived = onMsgReceived
});

我们每次注册消息,都要 new 一个 MsgRecord 对象出来,而我们在注销的时候,对这个对象是什么都没有做的,注销的代码如下:

foreach (var msgRecord in mMsgRecorder)
{MsgDispatcher.UnRegister(msgRecord.Name,msgRecord.OnMsgReceived);                
}

这样会造成一个性能问题,这个性能问题主要是有 new 时候寻址造成的,具体原因自行搜索,当然在本专栏的后边还是会介绍的。我们要做的,就是减少 new 的发生次数,要想减少,就得让我们的 MsgRecord 能够回收利用。

如何回收利用呢,答案是维护一个容器,比如 List 或者 Queue、Stack 等,也就是传说中的对象池。由于我们的 MsgRecord 的作用仅仅是作为一个存储结构而已,而存储的顺序也不是很重要,所以我们就用做简单的 Stack 结构,也就是栈,来作为 MsgRecord 对象池的容器。

其实现如下:

private class MsgRecord
{static Stack<MsgRecord> mMsgRecordPool = new Stack<MsgRecord>();public static MsgRecord Allocate(){if (mMsgRecordPool.Count > 0){return mMsgRecordPool.Pop();}return new MsgRecord();}public void Recycle(){Name = null;OnMsgReceived = null;mMsgRecordPool.Push(this);}public string Name;public Action<object> OnMsgReceived;
}

由于这个对象池只给 MsgRecord 用,所以就在 MsgRecord 内部实现了。
Allocate 是申请,也就是获取对象。Recycle 就是回收,当不用的时候调用一下就好了。

原理很简单。而 mMsgRecordPool 之所以设置成了 private 访问权限,是因为,不希望被外部访问到。对于一个类的设计来讲,MsgRecord 是一个非常合格的类了。

应用到我们的消息策略的代码如下:

protected void RegisterMsg(string msgName, Action<object> onMsgReceived)
{MsgDispatcher.Register(msgName, onMsgReceived);// var msgRecord = MsgRecord.Allocate();msgRecord.Name = msgName;msgRecord.OnMsgReceived = onMsgReceived;mMsgRecorder.Add(msgRecord);
}private void OnDestroy()
{OnBeforeDestroy();foreach (var msgRecord in mMsgRecorder){MsgDispatcher.UnRegister(msgRecord.Name,msgRecord.OnMsgReceived);  //msgRecord.Recycle();}mMsgRecorder.Clear();
}

我们发现,在申请对象部分可以简化成如下:

// var msgRecord = MsgRecord.Allocate();
//            
// msgRecord.Name = msgName;
// msgRecord.OnMsgReceived = onMsgReceived;
//            
// mMsgRecorder.Add(msgRecord);mMsgRecorder.Add(MsgRecord.Allocate(msgName, onMsgReceived));

只需要向 MsgRecord.Allocate 增加参数,代码如下:

public static MsgRecord Allocate(string msgName,Action<object> onMsgReceived)
{MsgRecord retMsgRecord = null;retMsgRecord = mMsgRecordPool.Count > 0 ? mMsgRecordPool.Pop() : new MsgRecord();retMsgRecord.Name = msgName;retMsgRecord.OnMsgReceived = onMsgReceived;return retMsgRecord;
}

代码不难,那么到这里,我们的完整的第十三个示例就写完了。

完整示例代码如下:

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;namespace QFramework
{public abstract partial class MonoBehaviourSimplify{List<MsgRecord> mMsgRecorder = new List<MsgRecord>();private class MsgRecord{private static readonly Stack<MsgRecord> mMsgRecordPool = new Stack<MsgRecord>();public static MsgRecord Allocate(string msgName,Action<object> onMsgReceived){MsgRecord retMsgRecord = null;retMsgRecord = mMsgRecordPool.Count > 0 ? mMsgRecordPool.Pop() : new MsgRecord();retMsgRecord.Name = msgName;retMsgRecord.OnMsgReceived = onMsgReceived;return retMsgRecord;}public void Recycle(){Name = null;OnMsgReceived = null;mMsgRecordPool.Push(this);}public string Name;public Action<object> OnMsgReceived;}protected void RegisterMsg(string msgName, Action<object> onMsgReceived){MsgDispatcher.Register(msgName, onMsgReceived);mMsgRecorder.Add(MsgRecord.Allocate(msgName, onMsgReceived));}private void OnDestroy(){OnBeforeDestroy();foreach (var msgRecord in mMsgRecorder){MsgDispatcher.UnRegister(msgRecord.Name,msgRecord.OnMsgReceived);  msgRecord.Recycle();}mMsgRecorder.Clear();}protected abstract void OnBeforeDestroy();}public class MsgDistapcherInMonoBehaviourSimplify : MonoBehaviourSimplify{
#if UNITY_EDITOR[UnityEditor.MenuItem("QFramework/13.消息机制集成到 MonoBehaviourSimplify", false, 14)]private static void MenuClicked(){UnityEditor.EditorApplication.isPlaying = true;new GameObject("MsgReceiverObj").AddComponent<MsgDistapcherInMonoBehaviourSimplify>();}
#endifprivate void Awake(){RegisterMsg("Do", DoSomething);RegisterMsg("Do", DoSomething);RegisterMsg("DO1", _ => { });RegisterMsg("DO2", _ => { });RegisterMsg("DO3", _ => { });}private IEnumerator Start(){MsgDispatcher.Send("Do","hello");yield return new WaitForSeconds(1.0f);MsgDispatcher.Send("Do","hello1");}void DoSomething(object data){// do somethingDebug.LogFormat("Received Do msg:{0}",data);}protected override void OnBeforeDestroy(){}}
}

运行结果如下图:
006tNc79gy1fzft5birfxj30w40ectar.jpg

菜单栏如下图:
006tNc79gy1fzft5ek7j9j30lk0fkamq.jpg

目录如下图:
006tNc79gy1fzft5i5y2ej30gm0dimz9.jpg

到这里我们可以进行一次导出了。

今天的内容就这些,我们下一篇在见,拜拜~

转载请注明地址:凉鞋的笔记:liangxiegame.com

更多内容

  • QFramework 地址:https://github.com/liangxiegame/QFramework

  • QQ 交流群:623597263

  • Unity 进阶小班

    • 主要训练内容:
      • 框架搭建训练(第一年)
      • 跟着案例学 Shader(第一年)
      • 副业的孵化(第二年、第三年)
    • 权益、授课形式等具体详情请查看《小班产品手册》:https://liangxiegame.com/master/intro
  • 关注公众号:liangxiegame 获取第一时间更新通知及更多的免费内容。

这篇关于Unity 游戏框架搭建 2019 (四十八) MonoBehaviourSimplify 中的消息策略完善的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/964417

相关文章

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Python 中 requests 与 aiohttp 在实际项目中的选择策略详解

《Python中requests与aiohttp在实际项目中的选择策略详解》本文主要介绍了Python爬虫开发中常用的两个库requests和aiohttp的使用方法及其区别,通过实际项目案... 目录一、requests 库二、aiohttp 库三、requests 和 aiohttp 的比较四、requ

Redis过期键删除策略解读

《Redis过期键删除策略解读》Redis通过惰性删除策略和定期删除策略来管理过期键,惰性删除策略在键被访问时检查是否过期并删除,节省CPU开销但可能导致过期键滞留,定期删除策略定期扫描并删除过期键,... 目录1.Redis使用两种不同的策略来删除过期键,分别是惰性删除策略和定期删除策略1.1惰性删除策略

Mycat搭建分库分表方式

《Mycat搭建分库分表方式》文章介绍了如何使用分库分表架构来解决单表数据量过大带来的性能和存储容量限制的问题,通过在一对主从复制节点上配置数据源,并使用分片算法将数据分配到不同的数据库表中,可以有效... 目录分库分表解决的问题分库分表架构添加数据验证结果 总结分库分表解决的问题单表数据量过大带来的性能

Java汇编源码如何查看环境搭建

《Java汇编源码如何查看环境搭建》:本文主要介绍如何在IntelliJIDEA开发环境中搭建字节码和汇编环境,以便更好地进行代码调优和JVM学习,首先,介绍了如何配置IntelliJIDEA以方... 目录一、简介二、在IDEA开发环境中搭建汇编环境2.1 在IDEA中搭建字节码查看环境2.1.1 搭建步

Python基于火山引擎豆包大模型搭建QQ机器人详细教程(2024年最新)

《Python基于火山引擎豆包大模型搭建QQ机器人详细教程(2024年最新)》:本文主要介绍Python基于火山引擎豆包大模型搭建QQ机器人详细的相关资料,包括开通模型、配置APIKEY鉴权和SD... 目录豆包大模型概述开通模型付费安装 SDK 环境配置 API KEY 鉴权Ark 模型接口Prompt

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

MyBatis框架实现一个简单的数据查询操作

《MyBatis框架实现一个简单的数据查询操作》本文介绍了MyBatis框架下进行数据查询操作的详细步骤,括创建实体类、编写SQL标签、配置Mapper、开启驼峰命名映射以及执行SQL语句等,感兴趣的... 基于在前面几章我们已经学习了对MyBATis进行环境配置,并利用SqlSessionFactory核

鸿蒙开发搭建flutter适配的开发环境

《鸿蒙开发搭建flutter适配的开发环境》文章详细介绍了在Windows系统上如何创建和运行鸿蒙Flutter项目,包括使用flutterdoctor检测环境、创建项目、编译HAP包以及在真机上运... 目录环境搭建创建运行项目打包项目总结环境搭建1.安装 DevEco Studio NEXT IDE

Python开发围棋游戏的实例代码(实现全部功能)

《Python开发围棋游戏的实例代码(实现全部功能)》围棋是一种古老而复杂的策略棋类游戏,起源于中国,已有超过2500年的历史,本文介绍了如何用Python开发一个简单的围棋游戏,实例代码涵盖了游戏的... 目录1. 围棋游戏概述1.1 游戏规则1.2 游戏设计思路2. 环境准备3. 创建棋盘3.1 棋盘类