为什么用消息机制?
三个字,解!!!!耦!!!!合!!!!。
我的框架中的消息机制用例:
1.接收者
using UnityEngine;
namespace QFramework.Example
- {
- ///
- /// 教程地址:http://liangxiegame.com/post/5/
- ///
- public class Receiver : MonoBehaviour, IMsgReceiver
- {
- // Use this for initialization
- private void Awake()
- {
- this.RegisterLogicMsg(“Receiver Show Sth”, ReceiveMsg);
}
private void ReceiveMsg(object[] args)
- {
- foreach (var arg in args)
- {
- Log.I(arg);
- }
- }
- }
- }
2.发送者
using UnityEngine;
namespace QFramework.Example
- {
- ///
- /// 教程地址:http://liangxiegame.com/post/5/
- ///
- public class Sender : MonoBehaviour, IMsgSender
- {
- void Update()
- {
- this.SendLogicMsg(“Receiver Show Sth”, “你好”, “世界”);
- }
- }
- }
3.运行结果
使用起来几行代码的事情,实现起来就没这么简单了。
如何实现的?
可以看到接收者实现了接口IMsgReceiver,发送者实现了接口IMsgSender。那先看下这两个接口定义。
IMsgReceiver:
- namespace QFramework
- {
- public interface IMsgReceiver
- {
- }
- }
IMsgSender
- namespace QFramework
- {
- public interface IMsgSender
- {
- }
- }
毛都没有啊。也没有SendLogicMsg或者ReceiveLogicMsg方法的定义啊。
答案是使用C# this的扩展方式实现接口方法。
不清楚的童鞋请百度C# this扩展,有好多文章就不介绍了。
以上先告一段落,先介绍个重要的角色,MsgDispatcher(消息分发器)。
贴上第一部分代码:
- ///
- /// 消息分发器
- /// C# this扩展 需要静态类
- /// 教程地址:http://liangxiegame.com/post/5/
- public static class MsgDispatcher
- {
- ///
- /// 消息捕捉器
- ///
- private class LogicMsgHandler
- {
- public readonly IMsgReceiver Receiver;
public readonly Action
public LogicMsgHandler(IMsgReceiver receiver, Action
- {
- Receiver = receiver;
- Callback = callback;
- }
}
///
- /// 每个消息名字维护一组消息捕捉器。
- ///
- static readonly Dictionary
> mMsgHandlerDict = - new Dictionary
>();
读注释!!!
贴上注册消息的代码
- ///
- /// 注册消息,
- /// 注意第一个参数,使用了C# this的扩展,
- /// 所以只有实现IMsgReceiver的对象才能调用此方法
- ///
- public static void RegisterLogicMsg(this IMsgReceiver self, string msgName, Action
- {
- // 略过
- if (string.IsNullOrEmpty(msgName))
- {
- Log.W(“RegisterMsg:” + msgName + “ is Null or Empty”);
- return;
}
// 略过
- if (null == callback)
- {
- Log.W(“RegisterMsg:” + msgName + “ callback is Null”);
- return;
}
// 略过
- if (!mMsgHandlerDict.ContainsKey(msgName))
- {
- mMsgHandlerDict[msgName] = new List
(); }
// 看下这里
var handlers = mMsgHandlerDict[msgName];
// 略过
- // 防止重复注册
- foreach (var handler in handlers)
- {
- if (handler.Receiver == self && handler.Callback == callback)
- {
- Log.W(“RegisterMsg:” + msgName + “ ayready Register”);
- return;
- }
}
// 再看下这里
- handlers.Add(new LogicMsgHandler(self, callback));
- }
为了节省您时间,略过部分的代码就不要看了,什么?!!你都看了!!!! 23333
发送消息相关的代码
- ///
- /// 发送消息
- /// 注意第一个参数
- ///
- public static void SendLogicMsg(this IMsgSender sender, string msgName, params object[] paramList)
- {
- // 略过,不用看
- if (string.IsNullOrEmpty(msgName))
- {
- Log.E(“SendMsg is Null or Empty”);
- return;
}
// 略过,不用看
- if (!mMsgHandlerDict.ContainsKey(msgName))
- {
- Log.W(“SendMsg is UnRegister”);
- return;
}
// 开始看!!!!
- var handlers = mMsgHandlerDict[msgName];
var handlerCount = handlers.Count;
// 之所以是从后向前遍历,是因为 从前向后遍历删除后索引值会不断变化
- // 参考文章,http://www.2cto.com/kf/201312/266723.html
- for (var index = handlerCount - 1; index >= 0; index—)
- {
var handler = handlers[index];
if (handler.Receiver != null)
- {
- Log.W(“SendLogicMsg:” + msgName + “ Succeed”);
- handler.Callback(paramList);
- }
- else
- {
- handlers.Remove(handler);
- }
- }
- }
OK主要的部分全都贴出来啦
可以改进的地方:
- 目前整个游戏的消息都由一个字典维护,可以改进为每个模块维护一个字典或者其他方式。
- 消息名字类型由字符串定义的,可以改成枚举转unsigned int方式。
- 欢迎补充。
坑
- 如果是MonoBehaviour注册消息之后,GameObject Destroy之前一定要注销消息,之前的解决方案是,自定义一个基类来维护该对象已经注册的消息列表,然后在基类的OnDestory时候遍历卸载。
- 欢迎补充。
相关链接:
我的框架地址:https://github.com/liangxiegame/QFramework
教程源码:https://github.com/liangxiegame/QFramework/tree/master/Assets/HowToWriteUnityGameFramework/
QFramework &游戏框架搭建QQ交流群: 623597263
转载请注明地址:凉鞋的笔记http://liangxiegame.com/
微信公众号:liangxiegame
如果有帮助到您:
如果觉得本篇教程对您有帮助,不妨通过以下方式赞助笔者一下,鼓励笔者继续写出更多高质量的教程,也让更多的力量加入 QFramework 。
- 给QFramework 一个 Star
- 给 Asset Store 上的 QFramework 并给个五星(需要先下载)
- 购买 gitchat 话题:《命名的力量:变量》
- 购买同名的蛮牛视频课程录播课程:
- 价格 49.2 元
- 地址:http://edu.manew.com/course/431
- 购买同名电子书:https://www.kancloud.cn/liangxiegame/unity_framework_design
- 价格 49.2 元,内容会在 2018 年 10 月份完结