skynet.rawcall使用详解及应用场景

news2025/4/12 15:02:45

目录

  • 核心特性
  • 函数原型
  • 使用场景
    • 场景 1:高性能二进制传输(如文件转发)
    • 场景 2:自定义序列化协议(如 Protocol Buffers)
    • 场景 3:跨服务共享内存(避免拷贝)
  • 配套接收方实现
  • 与 skynet.call 的对比
  • 注意事项
  • 典型错误示例
  • 总结

在这里插入图片描述

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


核心特性

  1. 非自动序列化

    • 直接传递 lightuserdata(C 指针)+ size(数据长度),不调用 skynet.pack/skynet.unpack
    • 适用于已序列化的二进制数据或需要避免序列化开销的场景。
  2. 同步调用

    • skynet.call 类似,发送请求后阻塞等待响应。
    • 返回值为接收方通过 skynet.ret 返回的原始数据指针(需手动处理)。
  3. 内存管理

    • 发送方和接收方需明确内存所有权,避免野指针或内存泄漏。

函数原型

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)。

使用场景

场景 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、结构化数据交互
错误处理需自行处理指针有效性框架自动捕获异常

注意事项

  1. 内存安全

    • 确保传递的指针在接收方使用期间有效。
    • 若数据由发送方分配,接收方不应释放;若需返回新数据,接收方应分配新内存。
  2. 协议一致性

    • 发送方和接收方必须使用相同的协议类型(typename)。
    • 接收方需正确注册协议处理函数(skynet.register_protocol)。
  3. 避免野指针

    • 使用 skynet.mallocskynet.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)  -- 确保接收方不再使用后释放

总结

使用场景优先级

  1. 高频二进制传输(如音视频流、日志批量处理)。
  2. 自定义序列化协议(如 Protobuf、FlatBuffers)。
  3. 零拷贝共享内存(大规模数据共享,避免复制开销)。
  4. 与 C 模块交互(直接传递 C 层分配的内存块)。

核心原则

  • 仅在必要时使用 skynet.rawcall,优先选择更安全的 skynet.call
  • 严格管理内存生命周期,结合 skynet.malloc/skynet.free 使用。
  • 确保发送方和接收方对数据格式和协议类型有明确约定。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2330953.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用SpringSecurity下,发生重定向异常

使用SpringSecurity下,发生空转异常 环境信息: Spring Boot 3.4.4 , jdk 17 , springSecurity 6.4.4 问题背景: 没有自定义controller ,改写了login 页面,并且进行了成功后的跳转处理&#xf…

Elasticsearch | ES索引模板、索引和索引别名的创建与管理

关注:CodingTechWork 引言 在使用 Elasticsearch (ES) 和 Kibana 构建数据存储和分析系统时,索引模板、索引和索引别名的管理是关键步骤。本文将详细介绍如何通过 RESTful API 和 Kibana Dev Tools 创建索引模板、索引以及索引别名,并提供具…

力扣hot100_回溯(2)_python版本

一、39. 组合总和(中等) 代码: class Solution:def combinationSum(self, candidates: List[int], target: int) -> List[List[int]]:ans []path []def dfs(i: int, left: int) -> None:if left 0:# 找到一个合法组合ans.append(pa…

LPDDR4内存颗粒命名规则全解析:三星、镁光、海力士、南亚、长鑫等厂商型号解码与选型指南

由于之前DDR的系列选型文章有很好的反馈,所以补充LPDDR4低功耗内存的选型和命名规则,总结了目前市面上常用的内存,供硬件工程师及数码爱好者参考。 在智能手机、平板电脑和低功耗设备中,LPDDR4 SDRAM凭借其高带宽、低功耗特性成为…

【杂谈】Godot4.4导出到Android平台(正式导出)

学博而后可约,事历而后知要。 目录 一、准备二、Gradle构建三、配置Java SDK四、配置Android SDK五、配置密钥 一、准备 本文在前文【杂谈】Godot4.4导出到安卓平台(调试导出)的基础上,进行正式导出。调试导出并不是真正的编译导…

基于AI设计开发出来的业务系统是什么样的?没有菜单?没有表格?

基于AI设计开发出的业务系统仍然会包含菜单、表格等传统UI元素,但AI技术会显著改变它们的实现方式和交互逻辑。以下是具体分析: 一、传统元素的持续存在 功能刚需性 • 菜单承担着系统导航的核心功能,表格则是结构化数据展示的基础载体。根…

数字足迹管理(DFM):你的网络隐身指南

数字足迹管理(DFM):你的网络隐身指南 ‌你可能不知道,你的姓名、电话、住址正在网上被“明码标价”‌ ——而这一切,可能只是因为你点过外卖、寄过快递,甚至注册过一个网站。 一、什么是数字足迹管理&#…

如何避免“过度承诺”导致的验收失败

如何避免“过度承诺”导致的验收失败?关键在于: 评估可行性、设置合理目标、高频沟通反馈、阶段性验收、做好风险管理。其中设置合理目标至关重要,很多团队往往在项目初期为迎合客户或领导而报出“最理想方案”,忽略了资源、技术及…

紧跟数字人热潮:123 数字人分身克隆系统源码部署与风口洞察

在当今数字化浪潮中,数字人技术无疑已成为最具活力与潜力的领域之一,正以迅猛之势席卷多个行业,重塑着人们的交互方式与商业运作模式。C 站作为技术交流的前沿阵地,汇聚了众多关注前沿科技的开发者与技术爱好者,今天来…

QT控件 修改QtTreePropertyBrowser自定义属性编辑器源码,添加第一列标题勾选,按钮,右键菜单事件等功能

头阵子遇到一个需要修改QtTreePropertyBrowser控件的需求,QT开发做这么久了,这个控件倒是第一次用,费了点时间研究,在这里做个简单的总结。 QtTreePropertyBrowser控件 是 Qt 解决方案 (Qt Solutions) 中的一个组件,用…

开源模型应用落地-模型上下文协议(MCP)-从数据孤岛到万物互联(一)

一、前言 当开发者还在为每个AI工具编写臃肿的API适配器时,一场关于「连接」的技术革命已悄然降临。模型上下文协议(MCP)正在用一套全新的交互语法,重新定义人工智能与物理世界的对话方式。MCP协议如同为AI系统装上了“万能接口”…

【区块链安全 | 第三十八篇】合约审计之获取私有数据(二)

文章目录 前言漏洞代码代码审计攻击步骤修复建议审计思路 前言 在【区块链安全 | 第三十七篇】合约审计之获取私有数据(一)中,介绍了私有数据、访问私有数据实例、Solidity 中的数据存储方式等知识,本文通过分析具体合约代码进行…

mac 苍穹外卖 后端初始 SkyApplication 报错

报错内容 java: java.lang.NoSuchFieldError: Class com.sun.tools.javac.tree.JCTree$JCImport does not have member field com.sun.tools.javac.tree.JCTree qualid deepseek 解决 打开 File > Project Structure > Project SDK, 选择 JDK17。我没有 JDK17就下载了一…

Proximal Policy Optimization (PPO)

2.1 策略梯度方法 策略梯度方法计算策略梯度的估计值并将其插入到随机梯度上升算法中。最常用的梯度估计器的形式如下: g ^ E t [ ∇ θ log ⁡ π θ ( a t ∣ s t ) A ^ t ] (1) \hat{g} \mathbb{E}_t \left[ \nabla_{\theta} \log \pi_{\theta}(a_t | s_t) \h…

微信小程序:动态表格实现,表头单元格数据完全从data中获取,宽度自定义,自定义文本框,行勾选,样式效果,横向滚动表格(解决背景色不足的问题)等

一、样式效果 二、代码 1、wxml <view class"line flex flex-center"><view class"none" wx:if"{{info.length 0}}">暂无料号</view><view wx:else class"table-container"><!-- 动态生成表头 -->&…

python-Leetcode 65.搜索旋转排序数组

题目&#xff1a; 整数数组nums按升序排列&#xff0c;数组中的值互不相同 在传递给函数之前&#xff0c;nums在预先未知的某个小标K上进行了旋转&#xff0c;使数组变为[nums[k], nums[k1], ..., nums[n-1], nums[0], nums[1], ..., nums[k-1]]&#xff0c;小标从0开始计数。…

Django学习记录-1

Django学习记录-1 虽然网上教程都很多&#xff0c;但是感觉自己记录一下才属于自己&#xff0c;之后想找也方面一点&#xff0c;文采不佳看的不爽可绕道。 参考贴 从零开始的Django框架入门到实战教程(内含实战实例) - 01 创建项目与app、加入静态文件、模板语法介绍&#xff…

K8s私有仓库拉取镜像报错解决:x509 certificate signed by unknown authority

前言 在Kubernetes环境中使用自签名证书的私有Harbor镜像仓库时&#xff0c;常会遇到证书验证失败的问题。本文将详细讲解如何解决这个常见的证书问题。 环境信息&#xff1a; Kubernetes版本&#xff1a;1.28.2容器运行时&#xff1a;containerd 1.6.20私有仓库&#xff1a…

LabVIEW 长期项目开发

LabVIEW 凭借其图形化编程的独特优势&#xff0c;在工业自动化、测试测量等领域得到了广泛应用。对于长期运行、持续迭代的 LabVIEW 项目而言&#xff0c;其开发过程涵盖架构设计、代码管理、性能优化等多个关键环节&#xff0c;每个环节都对项目的成功起着至关重要的作用。下面…

蓝桥杯真题——好数、R格式

目录 蓝桥杯2024年第十五届省赛真题-好数 【模拟题】 题目描述 输入格式 输出格式 样例输入 样例输出 提示 代码1&#xff1a;有两个案例过不了&#xff0c;超时 蓝桥杯2024年第十五届省赛真题-R 格式 【vector容器的使用】 题目描述 输入格式 输出格式 样例输入…