分析Lua观察者模式最佳实践之构建事件分发系统

 更新时间:2021年06月17日 09:15:15   作者:iwiniwin  
当对象间存在一对多关系时,则使用观察者模式(Observer Pattern)。比如,当一个对象被修改时,则会自动通知依赖它的对象。观察者模式属于行为型模式

一、前言

试想这样一个问题,当某个事件发生时,比如在游戏中A模块修改了用户的金币数,而B模块和C模块提供的功能都依赖于用户的金币数,那么,A模块在修改金币数的同时,就需要通知B模块和C模块。常规的方法就是A模块持有B模块和C模块的对象,然后分别通过调用对象接口的方式告诉它们,“嘿,我修改了用户的金币数,改成了10金币”。

但这样就带来了许多问题:

  • A模块引用了B模块和C模块,耦合严重
  • A模块修改金币数的方法中调用了B,C模块的方法,当这两个模块发生变化时(比如B模块接收金币数的接口名称改变了,或是C模块不再需要知道金币数改变了),A模块也要修改
  • 当又出现一个D模块也需要知道金币数的变化时,同样需要修改A模块以适应这种需求

为了解决上面的问题,我们自然想到了观察者模式。

二、观察者模式

这里简单说一下什么是观察者模式:定义对象之间的一对多依赖,这样一来,当一个对象改变状态时,它的所有依赖者(称之为观察者)都会接收到通知并自动更新。

观察者模式的好处是,对象之间是松耦合的,当一个对象改变状态时,它并不需要知道自己的观察者是谁,只需要发布通知即可。任何时候都可以增加或删除观察者,不会影响到发布通知的对象。而事件分发系统就是观察者模式的一个具体实现

三、事件分发系统

事件分发系统核心需要提供的功能主要包括以下几个部分:

  • 当一个对象发生改变时,可以认为此时产生了一个事件,提供一个派发事件的接口,以通知所有的观察者
  • 需要提供注册监听事件的接口,以让观察者可以订阅自己需要接收的事件
  • 还需提供反注册监听事件接口,以让观察者可以取消自己的订阅
  • 最好还能在订阅的时候设置优先级,优先级越高的可以越先被通知

四、使用事件分发系统解决问题

首先,来看看使用事件分发系统处理上面提到的问题,会是什么样的效果。

A模块只需要派发金币修改事件,B,C模块只需要订阅金币修改事件,之后便可以收到通知了。是不是很简单呢

local B = class()
function B:on_money_change( money )
    print(money, "B receive event")
end
-- 订阅金币修改事件
EventSystem:on(Event.MoneyChanged, B.on_money_change, {target = B})

local C = class()
function C:on_money_change( money )
    print(money, "C receive event")
end
EventSystem:on(Event.MoneyChanged, C.on_money_change, {target = C})
-- 在A模块中派发金币修改事件,当前金币为10
EventSystem:emit(Event.MoneyChanged, 10)

接下来会仔细解读一下这个EventSystem事件分发系统的Lua实现代码。

实现事件分发系统时,需要小心一些特殊情况,比如有以下几个坑,读者可以留意一下代码中对这几个坑的处理

  • 在事件派发的过程中订阅该事件,订阅还有优先级,需要小心处理排序问题
  • 在事件派发的过程中取消订阅该事件,需要采用标记移除,不能直接移除
  • 在事件派发的过程中又派发了该事件,如何确定事件派发完成

为了便于讲解,下面的代码省略了一些非关键性的代码,用--- ...代替。

五、注册监听事件接口

function EventSystem:on( event, func, params )
    --- ...
    local event_listener = self._listeners[event]
    params = params or {}
    local priority = params.priority or 0
    local target = params.target
    --- ...
    local cb = {target = target, func = func, id = id, priority = priority}
    table.insert(event_listener.list, cb)
    id = id + 1
    if priority > 0 then
        event_listener.need_sort = true
        self:sort(event_listener)
    end
end

on方法中event参数表示要注册监听的事件名称,func参数表示当事件发生时要触发的回调函数,params表示额外参数,可以设置注册监听的目标target(可以利用它反注册所有与其相关的监听),也可以设置要注册监听的优先级,优先级越高的越先执行。

on方法的实现还是比较简单的,主要就是将注册的相关信息插入到event_listener表中,但是明明注册的监听是有优先级的,却仍然只是调用table.insert将信息插入到表的末尾,这是为什么呢?读者可以先留意一下,后面会有详细解释。
还需要格外注意的是sort方法

function EventSystem:sort( listener )
    if listener.need_sort == true and listener.emit_count == 0 then
        table.sort(listener.list, function ( a, b )
            if a.priority == b.priority then
                return a.id < b.id
            else
                return a.priority > b.priority
            end
        end)
        listener.need_sort = false;
    end
end

可以看到sort方法必须在listener.emit_count == 0时才会进行排序,listener.emit_count == 0表示的是当前的事件没有处于派发状态,后面讲到派发接口时会详细解释,这里读者只需要知道其表示的含义即可。

事件处于派发状态时不能进行优先级排序原因是可能会造成回调的重复触发。

比如当前事件有4个回调 a, b, c, d,派发事件是顺序执行回调,当执行到第3个回调c时,如果在c回调中又注册了一个优先级最高的回调e,立刻排序的话,e插入到第一位,c会被挤到第4位,顺序执行到第4个回调时,导致c又被调用一次。

六、反注册事件监听接口

function EventSystem:off( event, func, params )
    --- ...
    local event_listener = self._listeners[event]
    params = params or {}
    for i,cb in ipairs(event_listener.list) do
        if cb.func == func and cb.target == params.target then
            if event_listener.emit_count > 0 then
                -- 派发过程中只进行标记删除
                cb.need_remove = true
                event_listener.need_clean = true
            else
                table.remove(event_listener.list, i)
            end
            break;
        end
    end
end

off方法用于取消事件监听,当事件未处于派发过程中时,直接调用table.remove移除注册信息即可,但当事件处于派发过程中时,不能直接移除,只能先进行标记。
在事件处于派发过程中时不能直接移除的原因是可能导致遗漏触发某些回调,比如当前事件有5个回调 a, b, c, d, e,顺序执行到第3个回调c时,如果在c回调中调用了off方法取消自己的监听,此时直接移除c的话,会导致d回调移动到第3位,e移动到第4位,顺序执行到第4个回调时,调用的是e而遗漏了d。

七、事件派发接口

function EventSystem:emit( event, ... )
    --- ...
    local event_listener = self._listeners[event]
    local interrupt = false
    local length = #event_listener.list
    -- 这里不能使用ipairs,确保不会触发在派发过程中注册的事件
    -- 只取当前已经注册的事件数量,如果在派发过程中再注册(调用了table.insert),本次派发也不会调用
    for i = 1, length do
        if interrupt == true then
            break
        end
        local cb = event_listener.list[i]
        if cb.func and cb.need_remove ~= true then
            event_listener.emit_count = event_listener.emit_count + 1
            if cb.target then
                interrupt = cb.func(cb.target, ...)
            else
                interrupt = cb.func(...)
            end
            event_listener.emit_count = event_listener.emit_count - 1
        end
    end
    self:sort(event_listener);
    self:clean(event_listener);
    return interrupt
end

emit方法负责派发一个事件,顺序执行event_listener中注册的回调。事件的派发支持中断,当执行某个回调时,如果这个回调返回了true则可以中断当前事件的派发。

值得一提的是,代码通过对应的event_listener.emit_count = event_listener.emit_count + 1event_listener.emit_count = event_listener.emit_count - 1来记录事件的派发状态,当emit_count > 0则表明事件还在派发过程中。当emit_count == 0则表明事件派发完成。

不能使用event_listener.is_emiting = trueevent_listener.is_emiting = false代替的原因是如果在触发的回调中又派发了事件,形成了递归,那么二次派发事件结束时会直接将event_listener.is_emiting置为flase,导致一次派发事件对应的派发状态被标记错误

八、更多

事件分发系统的完整源码可以点击这里查看,测试用例可以点击这里查看
更多Lua相关的设计与使用,比如面向对象(代码中用到的class关键字),组件系统,分模块加载等等,可以查看GitHub仓库LuaKit

以上就是分析Lua观察者模式最佳实践之构建事件分发系统的详细内容,更多关于Lua 观察者模式 构建事件分发系统的资料请关注脚本之家其它相关文章!

相关文章

  • Lua中实现StringBuffer功能

    Lua中实现StringBuffer功能

    这篇文章主要介绍了Lua中实现StringBuffer功能,本文给出了实现代码和调用代码,需要的朋友可以参考下
    2014-11-11
  • Lua中让回调函数支持回调对象方法的解决方法

    Lua中让回调函数支持回调对象方法的解决方法

    这篇文章主要介绍了Lua中让回调支持对象方法,一般情况下,Lua中只支持回调一个函数,本文方法实现可以回调一个对象的方法,需要的朋友可以参考下
    2014-12-12
  • Lua中的元表与元方法学习总结

    Lua中的元表与元方法学习总结

    这篇文章主要介绍了Lua中的元表与元方法学习总结,本文讲解了算术类的元方法、__tostring元方法等内容,需要的朋友可以参考下
    2014-09-09
  • Lua中的变量类型与语句学习总结

    Lua中的变量类型与语句学习总结

    这篇文章主要介绍了Lua中的变量类型与语句学习总结,总结了Lua入门过程中的一些基础知识,需要的朋友可以参考下
    2016-06-06
  • Lua教程(十七):C API简介

    Lua教程(十七):C API简介

    这篇文章主要介绍了Lua教程(十七):C API简介,本文讲解了基础知识、栈、C API中的错误处理、Lua调用C程序、C程序调用Lua代码的错误处理等内容,需要的朋友可以参考下
    2015-04-04
  • Lua中获取table长度问题探讨

    Lua中获取table长度问题探讨

    这篇文章主要介绍了Lua中获取table长度问题探讨,本文非常深入的研究了Lua中table长度的获取问题,分析了各种各样的情况,需要的朋友可以参考下
    2015-04-04
  • Lua模块与包学习笔记

    Lua模块与包学习笔记

    这篇文章主要介绍了Lua模块与包学习笔记,本文讲解了加载模块、加载机制等内容,需要的朋友可以参考下
    2014-12-12
  • lua中操作json数据的方法

    lua中操作json数据的方法

    这篇文章主要介绍了lua中操作json数据的方法,本文讲解使用cjson包操作JSON数据,并给出了操作实例,需要的朋友可以参考下
    2015-04-04
  • Lua教程(三):值与类型介绍

    Lua教程(三):值与类型介绍

    这篇文章主要介绍了Lua教程(三):值与类型介绍,本文起讲解了Lua的八种基本类型、userdata、thread、table等内容,需要的朋友可以参考下
    2015-03-03
  • Lua源码中字符串类型的实现

    Lua源码中字符串类型的实现

    与其他主流脚本语言不同的是,Lua在实现字符串类型有两方面不同。第一,所有的字符串在Lua中都只储存一份拷贝。第二,所有的字符串变量,只保存字符串引用,而不保存它的buffer。我们来具体看看lua源码中如何实现字符串类型的吧
    2015-04-04

最新评论