35人参与 • 2025-04-12 • 苹果IOS
skynet.rawcall
是 skynet 框架中用于直接传递原始二进制数据的低级通信接口,适用于需要绕过自动序列化/反序列化、手动控制内存或实现高性能传输的场景。以下是其详细用法和典型应用场景:
非自动序列化:
lightuserdata
(c 指针)+ size
(数据长度),不调用 skynet.pack
/skynet.unpack
。同步调用:
skynet.call
类似,发送请求后阻塞等待响应。skynet.ret
返回的原始数据指针(需手动处理)。内存管理:
local response_ptr, response_size = skynet.rawcall(target, typename, data_ptr, data_size)
target
:目标服务地址(如 skynet.self()
或服务句柄)。typename
:消息类型(字符串,需接收方注册对应的处理协议)。data_ptr
:原始数据指针(lightuserdata
)。data_size
:数据长度(number
)。response_ptr
:响应数据的指针(lightuserdata
)。response_size
:响应数据的长度(number
)。-- 发送方(直接传递文件内容指针) local file_content = read_file_as_binary("data.bin") local ptr, size = convert_to_lightuserdata(file_content) -- 假设已获得指针和大小 -- 同步调用目标服务,获取响应 local resp_ptr, resp_size = skynet.rawcall(target_service, "binary", ptr, size) -- 处理响应数据(需手动解析) process_response(resp_ptr, resp_size) skynet.free(resp_ptr) -- 手动释放响应内存(若由接收方分配)
-- 发送方(使用 protobuf 编码) local protobuf = require "protobuf" local msg = { id = 1001, name = "alice" } local encoded_data = protobuf.encode("myproto", msg) local ptr, size = get_data_pointer(encoded_data) -- 获取数据指针和长度 -- 发送原始数据并等待响应 local resp_ptr, resp_size = skynet.rawcall(target, "proto", ptr, size) local decoded_resp = protobuf.decode("responseproto", resp_ptr, resp_size) skynet.free(resp_ptr)
-- 发送方(传递共享内存指针) local shared_buf = skynet.malloc(1024) -- 分配共享内存 fill_buffer(shared_buf, 1024) -- 填充数据 -- 请求目标服务处理共享内存 local resp_ptr, resp_size = skynet.rawcall(target, "shared_mem", shared_buf, 1024) -- 处理完毕后释放内存 skynet.free(shared_buf) if resp_ptr ~= nil then skynet.free(resp_ptr) end
接收方需注册对应的协议类型,并手动处理原始数据指针:
-- 接收方服务 skynet.register_protocol { name = "binary", id = skynet.ptype_user, -- 自定义类型(如 100) unpack = function(ptr, size) return ptr, size end, -- 直接透传指针和大小 pack = function(ptr, size) return ptr, size end, -- 响应时不打包 } skynet.dispatch("binary", function(session, source, ptr, size) -- 处理原始数据 local result = process_binary_data(ptr, size) -- 返回响应(假设 result 是已分配的指针和大小) skynet.ret(result.ptr, result.size) end)
特性 | skynet.rawcall | skynet.call |
---|---|---|
数据传输 | 原始指针(无序列化) | 自动调用 skynet.pack /unpack |
性能 | 更高(避免序列化开销) | 较低(适合结构化数据) |
内存管理 | 需手动管理指针生命周期 | 框架自动管理 |
适用场景 | 大文件、自定义协议、共享内存 | 常规 rpc、结构化数据交互 |
错误处理 | 需自行处理指针有效性 | 框架自动捕获异常 |
内存安全:
协议一致性:
typename
)。skynet.register_protocol
)。避免野指针:
skynet.malloc
和 skynet.free
替代原生 malloc
/free
,确保内存池统一管理。-- 错误:传递临时栈指针(可能导致崩溃) local tmp_data = "hello" local ptr = get_pointer(tmp_data) skynet.rawcall(target, "test", ptr, #tmp_data) -- tmp_data 可能已被回收 -- 正确:分配堆内存并传递 local heap_ptr = skynet.malloc(1024) fill_data(heap_ptr) skynet.rawcall(target, "test", heap_ptr, 1024) skynet.free(heap_ptr) -- 确保接收方不再使用后释放
使用场景优先级:
核心原则:
skynet.rawcall
,优先选择更安全的 skynet.call
。skynet.malloc
/skynet.free
使用。到此这篇关于skynet.rawcall使用详解及应用场景的文章就介绍到这了,更多相关skynet.rawcall使用内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
您想发表意见!!点此发布评论
版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。
发表评论