Unity实用框架(三)事件系统

文章目录

  • Unity实用框架(三)事件系统
      • Event
      • IEventListener
      • MessageSystem
        • Subscribe/Unsubscribe
        • Publish

Unity实用框架(三)事件系统

在Unity提供的事件注册基础上,实现一个更加灵活的事件系统。

Event

首先,为所有可能的事件定义一个公用的事件基类。该基类包含:事件的发起者(UNITY对象)、要求响应的方式(同步或异步)以及事件是否在处理后销毁。

public abstract class Event : IDisposable
{
    private GameObject publisher;
    private bool responce;
    private bool disposeAfterUsed;
}

那么,任何类型只要继承自Event,就可以作为一个事件在MessageSystem里面进行发布和监听。当然,如果每种事件都单独写一个类型,或许会使得类的数量过多,可以将同模块下的所有事件封装到一个Event的子类当中,再通过字符串或枚举等方式进行二次查找。比如如下方式:

public class NetWorkConnectionEvent : Event{
    ...
    enum Type{
        EventType1,
        EventType2,
        ...
    } 
    ...
}

如果使用这种写法,MessageSystem会更加复杂,为了方便,我们在下面假设所有的事件都继承自Event,每个Event代表且仅代表一个事件。

IEventListener

一个类型实现该接口,以表明自己是一个可以监听事件的类型。该接口需要实现监听函数

public interface IEventListener
{
    bool OnEvent(Event e);
}

MessageSystem

Subscribe/Unsubscribe

一个基础的事件系统应当至少能够实现事件的发布和订阅功能。首先,我们定义以下几个列表,来储存订阅者信息。

private Dictionary> Listeners;
private List subscribeRequests;
private List > unsubscribeRequests

字典Listeners储存了注册到某个事件的所有监听者,用法很清晰,那么下面两个列表用处何在呢?由于场景中随时都会存在动态地事件注册和取消注册,因此,下面两个变量用于在MessageSystem.Update函数中,将新注册的Listener更新到Listener中,或从Listener中移除。当Listener的某个Key对应的Value为空时(即某个Event未被任何监听者注册),那么这个KEY将会被移除,相应地,而若新注册的监听者发现Listener中没有对应的Event,也会新增一个对应的Key。用这种动态的注册方法,可以避免事件系统中出现大量的无用注册信息以提高运行效率。

在Subscribe/Unsubscribe中,将需要Subscribe/Unsubscribe的监听者加入对应的列表。

subscribeRequests.Add(new KeyValuePair(e, listener);
unsubscribeRequests.Add(new KeyValuePair(e, listener);

在Update中,将它们更新到Listener:

foreach (var subReq in subscribeRequests)
{
    List callbacks = null;
    if (Listeners.TryGetValue(subReq.Key, out callbacks))
    {
        callbacks.Add(subReq.Value);
    }
    else
    {
        callbacks = new List();
        callbacks.Add(subReq.Value);
        listeners.Add(subReq.Key, callbacks);
    }
}
subscribeRequests.Clear();

var infoToUnsubscribe = new SubscribeCallbackInfo(null, false);
foreach (var unsubReq in unsubscribeRequests)
{
    List callbacks = null;
    if (Listeners.TryGetValue(unsubReq.Key, out callbacks))
    {
        if (callbacks.Remove(unsubReq.Value))
        {
            if (callbacks.Count == 0)
            {
                subscribeCallbacks.Remove(unsubReq.Key);
            }
        }
    }
}
unsubscribeRequests.Clear();

Publish

现在已经有了订阅信息,需要一个Publish函数来发布事件。

public void Publish(Event e);

发布者决定了事件的处理方式(同步-异步)以及销毁方式,对于异步调用,直接循环遍历Listener中对应的List即可:

List listeners;
if(Listeners.TryGetValue(e, out listeners)){
    for(listener in listeners){
        listener.OnEvent(e);
    }
}

对于同步调用,用StartCoroutine协程代替直接调用:

StartCoroutine(listener.OnEvent(e));

有些时候,我们只需要事件被任何一个监听者处理掉就够了,而不需要所有监听者都去处理它,因此,发布者可以将Event.disposeAfterUsed置为true,在publish中,只要某一个监听者处理了事件,那么销毁该事件,不需要将其向下传递。

List listeners;
if(Listeners.TryGetValue(e, out listeners)){
    for(listener in listeners){
        if(listener.OnEvent(e)){
            e.dispose();
            break;
        }
    }
}

更新时间:2022.7.17

你可能感兴趣的:(设计模式,unity实用框架,unity,c#,游戏引擎,设计模式,经验分享)