skynet.rawcall使用應(yīng)用場(chǎng)景分析
skynet.rawcall
是 Skynet 框架中用于直接傳遞原始二進(jìn)制數(shù)據(jù)的低級(jí)通信接口,適用于需要繞過(guò)自動(dòng)序列化/反序列化、手動(dòng)控制內(nèi)存或?qū)崿F(xiàn)高性能傳輸?shù)膱?chǎng)景。以下是其詳細(xì)用法和典型應(yīng)用場(chǎng)景:
核心特性
非自動(dòng)序列化:
- 直接傳遞
lightuserdata
(C 指針)+size
(數(shù)據(jù)長(zhǎng)度),不調(diào)用skynet.pack
/skynet.unpack
。 - 適用于已序列化的二進(jìn)制數(shù)據(jù)或需要避免序列化開(kāi)銷的場(chǎng)景。
同步調(diào)用:
- 與
skynet.call
類似,發(fā)送請(qǐng)求后阻塞等待響應(yīng)。 - 返回值為接收方通過(guò)
skynet.ret
返回的原始數(shù)據(jù)指針(需手動(dòng)處理)。
內(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
:消息類型(字符串,需接收方注冊(cè)對(duì)應(yīng)的處理協(xié)議)。data_ptr
:原始數(shù)據(jù)指針(lightuserdata
)。data_size
:數(shù)據(jù)長(zhǎng)度(number
)。
- 返回值:
response_ptr
:響應(yīng)數(shù)據(jù)的指針(lightuserdata
)。response_size
:響應(yīng)數(shù)據(jù)的長(zhǎng)度(number
)。
使用場(chǎng)景
場(chǎng)景 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ù)(需手動(dòng)解析) process_response(resp_ptr, resp_size) skynet.free(resp_ptr) -- 手動(dòng)釋放響應(yīng)內(nèi)存(若由接收方分配)
場(chǎng)景 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ù)指針和長(zhǎng)度 -- 發(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)
場(chǎng)景 3:跨服務(wù)共享內(nèi)存(避免拷貝)
-- 發(fā)送方(傳遞共享內(nèi)存指針) local shared_buf = skynet.malloc(1024) -- 分配共享內(nèi)存 fill_buffer(shared_buf, 1024) -- 填充數(shù)據(jù) -- 請(qǐng)求目標(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
配套接收方實(shí)現(xiàn)
接收方需注冊(cè)對(duì)應(yīng)的協(xié)議類型,并手動(dòng)處理原始數(shù)據(jù)指針:
-- 接收方服務(wù) skynet.register_protocol { name = "binary", id = skynet.PTYPE_USER, -- 自定義類型(如 100) unpack = function(ptr, size) return ptr, size end, -- 直接透?jìng)髦羔樅痛笮? pack = function(ptr, size) return ptr, size end, -- 響應(yīng)時(shí)不打包 } 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 的對(duì)比
特性 | skynet.rawcall | skynet.call |
---|---|---|
數(shù)據(jù)傳輸 | 原始指針(無(wú)序列化) | 自動(dòng)調(diào)用 skynet.pack /unpack |
性能 | 更高(避免序列化開(kāi)銷) | 較低(適合結(jié)構(gòu)化數(shù)據(jù)) |
內(nèi)存管理 | 需手動(dòng)管理指針生命周期 | 框架自動(dòng)管理 |
適用場(chǎng)景 | 大文件、自定義協(xié)議、共享內(nèi)存 | 常規(guī) RPC、結(jié)構(gòu)化數(shù)據(jù)交互 |
錯(cuò)誤處理 | 需自行處理指針有效性 | 框架自動(dòng)捕獲異常 |
注意事項(xiàng)
內(nèi)存安全:
- 確保傳遞的指針在接收方使用期間有效。
- 若數(shù)據(jù)由發(fā)送方分配,接收方不應(yīng)釋放;若需返回新數(shù)據(jù),接收方應(yīng)分配新內(nèi)存。
協(xié)議一致性:
- 發(fā)送方和接收方必須使用相同的協(xié)議類型(
typename
)。 - 接收方需正確注冊(cè)協(xié)議處理函數(shù)(
skynet.register_protocol
)。
避免野指針:
- 使用
skynet.malloc
和skynet.free
替代原生malloc
/free
,確保內(nèi)存池統(tǒng)一管理。
典型錯(cuò)誤示例
-- 錯(cuò)誤:傳遞臨時(shí)棧指針(可能導(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é)
使用場(chǎng)景優(yōu)先級(jí):
- 高頻二進(jìn)制傳輸(如音視頻流、日志批量處理)。
- 自定義序列化協(xié)議(如 Protobuf、FlatBuffers)。
- 零拷貝共享內(nèi)存(大規(guī)模數(shù)據(jù)共享,避免復(fù)制開(kāi)銷)。
- 與 C 模塊交互(直接傳遞 C 層分配的內(nèi)存塊)。
核心原則:
- 僅在必要時(shí)使用
skynet.rawcall
,優(yōu)先選擇更安全的skynet.call
。 - 嚴(yán)格管理內(nèi)存生命周期,結(jié)合
skynet.malloc
/skynet.free
使用。 - 確保發(fā)送方和接收方對(duì)數(shù)據(jù)格式和協(xié)議類型有明確約定。
到此這篇關(guān)于skynet.rawcall使用詳解及應(yīng)用場(chǎng)景的文章就介紹到這了,更多相關(guān)skynet.rawcall使用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
DVWA下載、安裝、使用(漏洞測(cè)試環(huán)境搭建)的詳細(xì)教程
這篇文章主要介紹了DVWA下載、安裝、使用(漏洞測(cè)試環(huán)境搭建)的詳細(xì)教程,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10Trie樹(shù)_字典樹(shù)(字符串排序)簡(jiǎn)介及實(shí)現(xiàn)
有時(shí),我們會(huì)碰到對(duì)字符串的排序,若采用一些經(jīng)典的排序算法,則時(shí)間復(fù)雜度一般為O(n*lgn),但若采用Trie樹(shù),則時(shí)間復(fù)雜度僅為O(n)2014-03-03聊聊Druid register mbean error的問(wèn)題
這篇文章主要介紹了Druid register mbean error的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-11-11gradle+shell實(shí)現(xiàn)自動(dòng)系統(tǒng)簽名
這篇文章主要介紹了gradle+shell實(shí)現(xiàn)自動(dòng)系統(tǒng)簽名的相關(guān)資料,需要的朋友可以參考下2019-08-08git pull時(shí)沖突的幾種解決方式(小結(jié))
這篇文章主要介紹了git pull時(shí)沖突的幾種解決方式,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07編程趣事:當(dāng)下流行編程語(yǔ)言的”討厭”程度排行榜
這篇文章主要介紹了編程趣事:當(dāng)下流行編程語(yǔ)言的”討厭”程度排行榜,和小編的感覺(jué)一樣,需要的朋友可以參考下2014-07-07chatgpt?1020?錯(cuò)誤碼成功解決的三種方案(推薦)
造成1020錯(cuò)誤的主要原因是代理問(wèn)題,當(dāng)打開(kāi)代理時(shí),登錄該網(wǎng)站會(huì)直接顯示上述錯(cuò)誤“Access?denied?Error?code?1020”,怎么解決這個(gè)問(wèn)題呢,下面小編給大家?guī)?lái)了chatgpt?1020?錯(cuò)誤碼成功解決的三種方案,感興趣的朋友一起看看吧2023-02-02