您现在的位置是:首页 >技术交流 >unity进阶学习笔记:消息框架网站首页技术交流

unity进阶学习笔记:消息框架

Raine_Yang 2024-06-03 11:54:19
简介unity进阶学习笔记:消息框架

1 使用消息框架的目的

对于小型游戏,可能不需要任何框架,而是让各个游戏脚本直接相互通信。如要实现玩家受到攻击血量减少,通过玩家控制类向血条脚本发送消息减少血量。但是这样直接通信会导致各脚本通信关系记为复杂,并且每一个脚本都和多个脚本有联系,导致维护和更新十分困难

我们利用上节课讲的管理类,可以将一类脚本由一个管理类控制,如将玩家的功能放在玩家管理类下,将血条,背包等UI组件放在UI管理类下。这样要减少玩家血量,可以让玩家控制类向UI管理类发消息,然后UI管理类来改变血条。该方案对大部分小型游戏适用,但对于大型游戏或者网游,一旦管理类过多依然难以管理

我们在管理类基础上再封装一层消息框架用于管理类间的通信。这样每一个类会和消息框架通信,由消息框架找到对于管理类,再由管理类控制对于脚本。
在这里插入图片描述

编写消息框架

1 消息类:用于保存要传输的消息类型及内容

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class Message
{
    public byte Type;
    public int Command;
    public object Content;

    public Message(byte type, int command, object content) {
        Type = type;
        Command = command;
        Content = content;
    }

}

这里我们的消息类定义三个参数Type, Command, Content。其中Type代表消息类型,这类大概很少,因此使用byte型保存即可以减少传输数据量。Command为具体的消息命令,使用int保存,之后可以定义不同值所对应的命令内容。Content为要传入的消息参数,类型不确定故设为object

为了明确各个类型和指令的具体含义,我们一般设置一个全局参数库用于存储各个指令名称和对应值,如下:

public class MessageType {
    // type
    public static byte Type_Audio = 1;
    public static byte Type_UI = 2;
    public static byte Type_Player = 3;
    // sound command
    public static int Audio_Play = 100;
    public static int Audio_Stop = 101;
    public static int Audio_Pause = 102;
    // UI command
    public static int UI_ShowPanal = 201;
    public static int UI_AddScore = 202;
    public static int UI_ShowShop = 203;
}

2 管理组件:

这里我们创建一个类MonoBase,里面多一个虚拟方法ReceiveMessage用于接受消息。之后的组件程序全部继承MonoBase类实现消息接受功能

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class MonoBase : MonoBehaviour
{

    // override in child class
    public virtual void ReceiveMessage(Message message) {

    }

    // Start is called before the first frame update
    void Start()
    {
        
    }

    // Update is called once per frame
    void Update()
    {
        
    }
}
using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public abstract class ManagerBase : SingletonBase<ManagerBase>
{

    // store all components
    public List<MonoBase> Monos = new List<MonoBase>();

    // register a component onto the manager
    public void Register(MonoBase mono) {
        if (!Monos.Contains(mono)) {
            Monos.Add(mono);
        }
    }

}

3 管理器:

管理器用于管理各个组件。这里我们创建的管理基类要实现以下方法:

1 存储其下所有的组件
2 让新组件可以注册进管理类
3 接收由消息管理器发布的消息,如果消息类型匹配将消息向下传递到各组件

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public abstract class ManagerBase : SingletonBase<ManagerBase>
{

    // store all components
    public List<MonoBase> Monos = new List<MonoBase>();

    // register a component onto the manager
    public void Register(MonoBase mono) {
        if (!Monos.Contains(mono)) {
            Monos.Add(mono);
        }
    }


    public virtual void ReceiveMessage(Message message) {
        // discard unmatched message type
        if (message.Type != GetMessageType()) {
            return;
        }
        foreach (var mono in Monos) {
            mono.ReceiveMessage(message);
        }
    }

    public abstract byte GetMessageType();

}

4 消息中心:

消息中心用于管理各个管理类,因此消息中心也具有保存管理类列表和注册新管理类的作用。消息类作为消息系统发送端,用于向各个管理类发送信息,管理类会自动屏蔽非本类型的消息(实现在上面管理类代码)

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class MessageCenter : MonoBehaviour
{

    // store all managers
    public List<ManagerBase> Managers = new List<ManagerBase>();

    // register a new manager
    public void Register(ManagerBase manager) {
        if (!Managers.Contains(manager)) {
            Managers.Add(manager);
        }
    }

    // send message
    public void SendCustomMessage(Message message) {
        foreach (var manager in Managers) {
            manager.ReceiveMessage(message);
        }
    }
}

至此整个消息框架搭建完毕

在下面我们使用一个简单的吃金币游戏来使用消息框架

1 搭建游戏场景
在这里插入图片描述
创建一个平面,加上灰色材质

创建一个胶囊体作为角色,添加上红色材质,RigidBody组件(设置为Kinetic并冻结旋转),将tag设为Player

创建圆柱体作为金币,添加黄色材质,将碰撞器设为Trigger,新建tag Coin并将金币物体加上Coin tag。复制多份金币

创建UI Canvas,并在左上角加上Panel,Panal锚点放在屏幕左上角。在Panel上加上(旧版)Text用于显示分数

创建空物体用于挂载MessageCenter

2 UIManager脚本

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class UIManager : ManagerBase
{
    // Start is called before the first frame update
    void Start()
    {
        // register to message center
        MessageCenter.Instance.Register(this);
    }

    // Update is called once per frame
    void Update()
    {
        
    }

    public override byte GetMessageType() {
        return MessageType.Type_UI;
    }
}

UIManager继承ManagerBase。在初始化UIManager时将其注册到MessageCenter里。这么我们重写ManagerBase里的GetMessageType,让其返回MessageType.Type_UI。ReceiveMessage方法不需要重写

3 Panal脚本

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;

public class Panal : MonoBase
{

    public Text text;

    // Start is called before the first frame update
    void Start()
    {
        UIManager.Instance.Register(this);
    }

    // Update is called once per frame
    void Update()
    {
        
    }

    public override void ReceiveMessage(Message message) {
        base.ReceiveMessage(message);
        if (message.Command == MessageType.UI_AddScore) {
            int score = (int)message.Content;
            // display score
            text.text = "score: " + score;
        }
    }
}

Panal用于控制积分表。在初始化时在UIManager中注册。在ReceiveMessage方法里调用基类的ReceiveMessage得到MessageCenter里发布的消息

3 PlayerManager脚本

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class PlayerManager : MonoBehaviour
{

    int score = 0;

    // Start is called before the first frame update
    void Start()
    {
        
    }

    // Update is called once per frame
    void Update()
    {
        float horizontal = Input.GetAxis("Horizontal");
        float vertical = Input.GetAxis("Vertical");
        Vector3 dir = new Vector3(horizontal, 0, vertical);
        if (dir != Vector3.zero) {
            transform.Translate(dir * 5 * Time.deltaTime);
        }
    }

    private void OnTriggerEnter(Collider other) {
        if (other.tag == "Coin") {
            score += 1;
            Destroy(other.gameObject);
            // send message
            MessageCenter.Instance.SendCustomMessage(new Message(MessageType.Type_UI, MessageType.UI_AddScore, score));
        }
    }

}

角色控制部分都是unity很基础内容,不额外说明。在该类里调用MessageCenter发布了玩家分数score

总结整个通信流程:
1 PlayerManager类调用MessageCenter发布Type_UI类型,UI_AddScore指令,数据为score

2 MessageCenter调用各个管理器的ReceiveMessage方法,其中Type_UI类型信息被UIManager类获取返回给子类Panal的ReceiveMessage

3 ReceiveMessage得到信息,在UI上显示出分数

风语者!平时喜欢研究各种技术,目前在从事后端开发工作,热爱生活、热爱工作。