it编程 > App开发 > 苹果IOS

skynet.rawcall使用应用场景分析

35人参与 2025-04-12 苹果IOS

skynet.rawcall 是 skynet 框架中用于直接传递原始二进制数据的低级通信接口,适用于需要绕过自动序列化/反序列化、手动控制内存或实现高性能传输的场景。以下是其详细用法和典型应用场景:

核心特性

非自动序列化

同步调用

内存管理

函数原型

local response_ptr, response_size = skynet.rawcall(target, typename, data_ptr, data_size)

使用场景

场景 1:高性能二进制传输(如文件转发)

-- 发送方(直接传递文件内容指针)
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)  -- 手动释放响应内存(若由接收方分配)

场景 2:自定义序列化协议(如 protocol buffers)

-- 发送方(使用 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)

场景 3:跨服务共享内存(避免拷贝)

-- 发送方(传递共享内存指针)
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.call 的对比

特性skynet.rawcallskynet.call
数据传输原始指针(无序列化)自动调用 skynet.pack/unpack
性能更高(避免序列化开销)较低(适合结构化数据)
内存管理需手动管理指针生命周期框架自动管理
适用场景大文件、自定义协议、共享内存常规 rpc、结构化数据交互
错误处理需自行处理指针有效性框架自动捕获异常

注意事项

内存安全

协议一致性

避免野指针

典型错误示例

-- 错误:传递临时栈指针(可能导致崩溃)
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.rawcall使用内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)
打赏 微信扫一扫 微信扫一扫

您想发表意见!!点此发布评论

推荐阅读

skynet.call使用详细解析

04-12

skynet.dispatch 使用示例详解

04-12

鸿蒙中Axios数据请求的封装和配置方法

04-12

基带版本降级了! 苹果 iOS 18.5 开发者预览版 Beta 发布

04-03

淘宝上的视频怎么保存到手机里面?具体如何操作?

03-29

苹果iOS18.2.1正式版紧急发布: 修复4大项Bug建议立刻安装

01-09

猜你喜欢

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论