skynet.rawcall使用應(yīng)用場景分析
skynet.rawcall
是 Skynet 框架中用于直接傳遞原始二進(jìn)制數(shù)據(jù)的低級通信接口,適用于需要繞過自動序列化/反序列化、手動控制內(nèi)存或?qū)崿F(xiàn)高性能傳輸?shù)膱鼍?。以下是其詳?xì)用法和典型應(yīng)用場景:
核心特性
非自動序列化:
- 直接傳遞
lightuserdata
(C 指針)+size
(數(shù)據(jù)長度),不調(diào)用skynet.pack
/skynet.unpack
。 - 適用于已序列化的二進(jìn)制數(shù)據(jù)或需要避免序列化開銷的場景。
同步調(diào)用:
- 與
skynet.call
類似,發(fā)送請求后阻塞等待響應(yīng)。 - 返回值為接收方通過
skynet.ret
返回的原始數(shù)據(jù)指針(需手動處理)。
內(nèi)存管理:
- 發(fā)送方和接收方需明確內(nèi)存所有權(quán),避免野指針或內(nèi)存泄漏。
函數(shù)原型
local response_ptr, response_size = skynet.rawcall(target, typename, data_ptr, data_size)
- 參數(shù):
target
:目標(biāo)服務(wù)地址(如skynet.self()
或服務(wù)句柄)。typename
:消息類型(字符串,需接收方注冊對應(yīng)的處理協(xié)議)。data_ptr
:原始數(shù)據(jù)指針(lightuserdata
)。data_size
:數(shù)據(jù)長度(number
)。
- 返回值:
response_ptr
:響應(yīng)數(shù)據(jù)的指針(lightuserdata
)。response_size
:響應(yīng)數(shù)據(jù)的長度(number
)。
使用場景
場景 1:高性能二進(jìn)制傳輸(如文件轉(zhuǎn)發(fā))
-- 發(fā)送方(直接傳遞文件內(nèi)容指針) local file_content = read_file_as_binary("data.bin") local ptr, size = convert_to_lightuserdata(file_content) -- 假設(shè)已獲得指針和大小 -- 同步調(diào)用目標(biāo)服務(wù),獲取響應(yīng) local resp_ptr, resp_size = skynet.rawcall(target_service, "binary", ptr, size) -- 處理響應(yīng)數(shù)據(jù)(需手動解析) process_response(resp_ptr, resp_size) skynet.free(resp_ptr) -- 手動釋放響應(yīng)內(nèi)存(若由接收方分配)
場景 2:自定義序列化協(xié)議(如 Protocol Buffers)
-- 發(fā)送方(使用 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) -- 獲取數(shù)據(jù)指針和長度 -- 發(fā)送原始數(shù)據(jù)并等待響應(yīng) 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:跨服務(wù)共享內(nèi)存(避免拷貝)
-- 發(fā)送方(傳遞共享內(nèi)存指針) local shared_buf = skynet.malloc(1024) -- 分配共享內(nèi)存 fill_buffer(shared_buf, 1024) -- 填充數(shù)據(jù) -- 請求目標(biāo)服務(wù)處理共享內(nèi)存 local resp_ptr, resp_size = skynet.rawcall(target, "shared_mem", shared_buf, 1024) -- 處理完畢后釋放內(nèi)存 skynet.free(shared_buf) if resp_ptr ~= nil then skynet.free(resp_ptr) end
配套接收方實現(xiàn)
接收方需注冊對應(yīng)的協(xié)議類型,并手動處理原始數(shù)據(jù)指針:
-- 接收方服務(wù) 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, -- 響應(yīng)時不打包 } skynet.dispatch("binary", function(session, source, ptr, size) -- 處理原始數(shù)據(jù) local result = process_binary_data(ptr, size) -- 返回響應(yīng)(假設(shè) result 是已分配的指針和大?。? skynet.ret(result.ptr, result.size) end)
與 skynet.call 的對比
特性 | skynet.rawcall | skynet.call |
---|---|---|
數(shù)據(jù)傳輸 | 原始指針(無序列化) | 自動調(diào)用 skynet.pack /unpack |
性能 | 更高(避免序列化開銷) | 較低(適合結(jié)構(gòu)化數(shù)據(jù)) |
內(nèi)存管理 | 需手動管理指針生命周期 | 框架自動管理 |
適用場景 | 大文件、自定義協(xié)議、共享內(nèi)存 | 常規(guī) RPC、結(jié)構(gòu)化數(shù)據(jù)交互 |
錯誤處理 | 需自行處理指針有效性 | 框架自動捕獲異常 |
注意事項
內(nèi)存安全:
- 確保傳遞的指針在接收方使用期間有效。
- 若數(shù)據(jù)由發(fā)送方分配,接收方不應(yīng)釋放;若需返回新數(shù)據(jù),接收方應(yīng)分配新內(nèi)存。
協(xié)議一致性:
- 發(fā)送方和接收方必須使用相同的協(xié)議類型(
typename
)。 - 接收方需正確注冊協(xié)議處理函數(shù)(
skynet.register_protocol
)。
避免野指針:
- 使用
skynet.malloc
和skynet.free
替代原生malloc
/free
,確保內(nèi)存池統(tǒng)一管理。
典型錯誤示例
-- 錯誤:傳遞臨時棧指針(可能導(dǎo)致崩潰) local tmp_data = "Hello" local ptr = get_pointer(tmp_data) skynet.rawcall(target, "test", ptr, #tmp_data) -- tmp_data 可能已被回收 -- 正確:分配堆內(nèi)存并傳遞 local heap_ptr = skynet.malloc(1024) fill_data(heap_ptr) skynet.rawcall(target, "test", heap_ptr, 1024) skynet.free(heap_ptr) -- 確保接收方不再使用后釋放
總結(jié)
使用場景優(yōu)先級:
- 高頻二進(jìn)制傳輸(如音視頻流、日志批量處理)。
- 自定義序列化協(xié)議(如 Protobuf、FlatBuffers)。
- 零拷貝共享內(nèi)存(大規(guī)模數(shù)據(jù)共享,避免復(fù)制開銷)。
- 與 C 模塊交互(直接傳遞 C 層分配的內(nèi)存塊)。
核心原則:
- 僅在必要時使用
skynet.rawcall
,優(yōu)先選擇更安全的skynet.call
。 - 嚴(yán)格管理內(nèi)存生命周期,結(jié)合
skynet.malloc
/skynet.free
使用。 - 確保發(fā)送方和接收方對數(shù)據(jù)格式和協(xié)議類型有明確約定。
到此這篇關(guān)于skynet.rawcall使用詳解及應(yīng)用場景的文章就介紹到這了,更多相關(guān)skynet.rawcall使用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
DVWA下載、安裝、使用(漏洞測試環(huán)境搭建)的詳細(xì)教程
這篇文章主要介紹了DVWA下載、安裝、使用(漏洞測試環(huán)境搭建)的詳細(xì)教程,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-10-10聊聊Druid register mbean error的問題
這篇文章主要介紹了Druid register mbean error的問題,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-11-11gradle+shell實現(xiàn)自動系統(tǒng)簽名
這篇文章主要介紹了gradle+shell實現(xiàn)自動系統(tǒng)簽名的相關(guān)資料,需要的朋友可以參考下2019-08-08