31人参与 • 2025-04-12 • 苹果IOS
skynet.dispatch
是 skynet 框架中用于注册消息处理函数的核心 api。它的作用是为特定类型的消息绑定处理逻辑,当服务收到该类型的消息时,自动调用对应的处理函数。以下是其详细解析:
skynet.dispatch(type, func)
type
:消息类型(字符串或数字)。
"lua"
:默认的 lua 消息协议(对应 skynet.ptype_lua
)。"socket"
:网络消息(对应 skynet.ptype_socket
)。skynet.register_protocol
注册的协议类型。func
:消息处理函数,格式为 function(session, source, ...)
。session
:会话 id,用于响应消息(如 skynet.ret
)。source
:发送方服务的地址(skynet.address
格式)。...
:消息内容(由协议定义的解包逻辑生成)。当服务收到一条消息时,skynet 会执行以下步骤:
"lua"
)找到对应的解包函数。unpack
函数,将二进制数据解析为 lua 值。skynet.dispatch
注册的处理函数,传入 session
、source
和解包后的数据。local skynet = require "skynet" -- 注册 lua 类型消息的处理函数 skynet.dispatch("lua", function(session, source, cmd, ...) if cmd == "add" then local a, b = ... skynet.ret(skynet.pack(a + b)) elseif cmd == "ping" then skynet.ret(skynet.pack("pong")) end end) skynet.start(function() -- 服务初始化代码 end)
说明:
"lua"
类型的消息时,解析出命令 cmd
和参数。cmd
执行逻辑,并通过 skynet.ret
返回结果。local skynet = require "skynet" -- 注册自定义协议 skynet.register_protocol { name = "myproto", id = 100, -- 自定义协议 id(需唯一) unpack = function(msg, sz) -- 自定义解包逻辑(如 sproto 解析) return myproto.decode(msg, sz) end } -- 处理自定义协议消息 skynet.dispatch("myproto", function(session, source, data) print("received:", data) skynet.ret() -- 无返回值 end) skynet.start(function() -- 服务初始化代码 end)
skynet.register_protocol
注册。"myproto"
的消息时,调用自定义解包函数,并处理数据。skynet.call
、skynet.sleep
等阻塞 api,当前协程会被挂起,直到操作完成。skynet.dispatch("lua", function(session, source, cmd) if cmd == "slow_task" then skynet.sleep(100) -- 挂起协程 1 秒 skynet.ret("done") end end)
skynet.ret
:用于向发送方返回响应。
skynet.call
),必须调用 skynet.ret
。skynet.send
),无需返回。skynet.dispatch("lua", function(session, source, cmd) if session ~= 0 then -- 需要响应 skynet.ret(skynet.pack("response")) end end)
协议注册:定义如何解析和打包消息。
skynet.register_protocol { name = "binary", id = skynet.ptype_user, -- 自定义 id unpack = function(msg, sz) return msg, sz end, -- 不解包,直接传递原始数据 pack = skynet.pack -- 默认打包函数 }
消息分发:通过 skynet.dispatch
绑定处理逻辑。
skynet.dispatch("binary", function(session, source, msg, sz) -- 处理二进制数据 end)
避免阻塞主线程:
若处理函数中有耗时操作(如大量计算、同步 io),应使用 skynet.fork
创建新协程。
skynet.dispatch("lua", function(session, source, cmd) if cmd == "heavy_task" then skynet.fork(function() -- 在子协程中执行耗时操作 local result = heavy_compute() skynet.ret(skynet.pack(result)) end) end end)
协程生命周期:
确保每个协程最终退出,避免内存泄漏(如通过 pcall
捕获异常)。
线程安全:
skynet 服务是单线程的,但协程间共享 lua 虚拟机状态,需谨慎处理共享数据(推荐使用 skynet.sharedata
)。
skynet.timeout
触发延时逻辑。skynet.dispatch
是 skynet 服务的消息处理入口,通过绑定协议类型与处理函数,实现灵活的消息分发机制。理解其协程调度、协议注册和响应机制,是构建高效 skynet 服务的关键。结合 sproto
等协议工具,可以进一步简化网络通信的复杂性。
到此这篇关于skynet.dispatch 使用示例详解的文章就介绍到这了,更多相关skynet.dispatch 使用内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
您想发表意见!!点此发布评论
版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。
发表评论