使用 Lua 协程处理异步回调函数

异步回调

项目使用 C++ + Lua 方式,C++ 主要编写一些底层模块,通常提供异步的方式

Lua 脚本中,使用这些 C++ 导入的模块,和异步方式编写代码

一般的一次异步调用,通常还可以编写逻辑清晰的代码

但是会有些功能会比较复杂的,一段逻辑内,会多次读写 Redis 等

使得搬砖异常吃力

回调例子

假设,我们一个这样的函数:

---@class c_model
---@field c_foo fun(_callback:fun( _ok:boolean, _result:string))

c_model.c_foo 接受一个参数 _callback 回调函数

现在希望 c_model.c_foo 可以按顺序的方式编写代码

1. 封装启动一个协程

---go 函数创建并启动一个协程
---@param _co_task function @函数原型 fun(_co:thread)
function go(_co_task)
    local co = coroutine.create(_co_task) -- 创建的协程是暂停的
    coroutine.resume(co, co)              -- 调用 coroutine.resume 激活协程执行
end

这里定义了一个 go 函数,接收fun(_co:thread)类型的函数

可以让这个函数以协程的方式运行

2. 封装 c_model.c_foo 使之成为协程函数

---封装 c_model.c_foo 异步函数,成为协程函数
---@param _co thread @协程对象
---@return boolean,string
function co_foo(_co)
    c_model.c_foo(function(_ok, _result)
        coroutine.resume(_co, _ok, _result) -- 2. 回调函数被调用后,激活本协程继续执行。并把_ok, _result传递给 yield
    end)
    return coroutine.yield()                -- 1. 主动放弃运行,本协程被切换出去
end

co_foo 函数内:

  1. coroutine.yield 先会被执行,并切出
  2. coroutine.resume , 在回调函数被执行时,会执行,并传递结果给 yield
  3. yield 这里重新被切进来, return 结果返回

使用协程,顺序编写代码的例子

---test顺序编写代码,解决回调函数造成同块逻辑被撕裂的例子
---@param _co thread @协程对象
function test(_co)
    for i = 1, 10, 1 do
        local ok, result = co_foo(_co) -- co_foo 会先 yield 切出;内部回调被执行时, resume 重新切回来继续执行
        print(ok, result)
    end
end

-- 启动 test 协程
go(test)

总结

  • 通过封装提供 go 函数,可以像 Go 语言一样,提供简易的启动协程界面
  • 项目的所有异步均可以封装提供 co_xxx 函数,使得需要回调的 API 均可以顺序编写代码。例如
    • co_sleep
    • co_rpc_call
    • co_redis_get / co_redis_set …
    • co_mysql_query
      等等
  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

fananchong2

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值