溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

怎么在Lua中管理C對(duì)象

發(fā)布時(shí)間:2021-07-24 15:16:21 來(lái)源:億速云 閱讀:127 作者:chen 欄目:web開(kāi)發(fā)

本篇內(nèi)容主要講解“怎么在Lua中管理C對(duì)象”,感興趣的朋友不妨來(lái)看看。本文介紹的方法操作簡(jiǎn)單快捷,實(shí)用性強(qiáng)。下面就讓小編來(lái)帶大家學(xué)習(xí)“怎么在Lua中管理C對(duì)象”吧!

今天同事在設(shè)計(jì)引擎的腳本接口時(shí)遇到一個(gè)問(wèn)題:需要把 C 對(duì)象指針?lè)诺?Lua 中,允許 Lua  保存這個(gè)指針,并傳遞給其它模塊。 這是給 Lua 寫(xiě) C 擴(kuò)展時(shí)常見(jiàn)的問(wèn)題,撇開(kāi)如何如何將對(duì)象的方法導(dǎo)入 Lua  這個(gè)更復(fù)雜的問(wèn)題不談,我主要想說(shuō)說(shuō) C 對(duì)象的生命期管理的問(wèn)題。 一開(kāi)始的設(shè)計(jì)是把對(duì)象的銷毀方法也導(dǎo)入 Lua  ,由腳本程序員手工管理。這是很明顯的 C 程序員的思路:誰(shuí)構(gòu)造誰(shuí)釋放。但在這里是不合適的,不符合帶 gc 機(jī)制語(yǔ)言的習(xí)慣。 ... Click  to expand...

今天同事在設(shè)計(jì)引擎的腳本接口時(shí)遇到一個(gè)問(wèn)題:需要把 C 對(duì)象指針?lè)诺?Lua 中,允許 Lua 保存這個(gè)指針,并傳遞給其它模塊。

這是給 Lua 寫(xiě) C 擴(kuò)展時(shí)常見(jiàn)的問(wèn)題,撇開(kāi)如何如何將對(duì)象的方法導(dǎo)入 Lua 這個(gè)更復(fù)雜的問(wèn)題不談,我主要想說(shuō)說(shuō) C 對(duì)象的生命期管理的問(wèn)題。

一開(kāi)始的設(shè)計(jì)是把對(duì)象的銷毀方法也導(dǎo)入 Lua ,由腳本程序員手工管理。這是很明顯的 C 程序員的思路:誰(shuí)構(gòu)造誰(shuí)釋放。但在這里是不合適的,不符合帶 gc 機(jī)制語(yǔ)言的習(xí)慣。

我們當(dāng)然希望腳本更為健壯,不需要考慮對(duì)象釋放的問(wèn)題。所以晚上我想了一下,修改了一下這部分的實(shí)現(xiàn)。

從效率方面著手,這個(gè)問(wèn)題分兩種情況:

***種情況很簡(jiǎn)單,C 對(duì)象可以被傳入 Lua 狀態(tài)機(jī)后,邏輯上可以確保它的指針一定一直有效,程序直到 Lua 狀態(tài)機(jī)本身關(guān)閉后,才會(huì)刪除對(duì)象。這種情況我們只需要把 C 對(duì)象指針以 lightuserdata 的形式壓入堆棧即可。

第二種情況就是,C 對(duì)象由腳本創(chuàng)建或獲得。在沒(méi)有地方對(duì)其引用之后,對(duì)象則應(yīng)該被刪除以釋放其占用的資源。這種情況,我們應(yīng)該使用 fulluserdata ,為其注冊(cè) gc 元方法。

不過(guò)問(wèn)題復(fù)雜在,引用 C 對(duì)象的可以是腳本也可以在 C 代碼中。腳本中對(duì) userdata 的引用 lua 狀態(tài)機(jī)會(huì)自行解決,但 lua 的 gc 過(guò)程并不能直接知道 C 中是否對(duì)對(duì)象還有引用,這就是我們需要做的工作了。

python 的 C 接口提供了相關(guān)的函數(shù),可以在 C 界面上對(duì) PyObject 加減引用。但是 lua 的 gc 是基于根掃描的,狀態(tài)機(jī)中并沒(méi)有引用計(jì)數(shù)。很自然的,lua 就沒(méi)有類似的 C 接口了。

我的解決方法是,在 lua 注冊(cè)表中創(chuàng)建一個(gè)弱表(value 是弱的,而 key 是強(qiáng)的),把 C 對(duì)象指針和對(duì)應(yīng)的  fulluserdata 以及它在 C 中的引用數(shù)量記入這個(gè)表里。然后提供一對(duì) API 對(duì)引用計(jì)數(shù)增減。當(dāng)引用計(jì)數(shù)為 0  時(shí),清除關(guān)于計(jì)數(shù)的表項(xiàng)。最終可利用 gc 回收掉已無(wú)引用的 C 對(duì)象。

詳細(xì)的程序可以參考我的 wiki 上貼的代碼。

這里補(bǔ)充幾點(diǎn)說(shuō)明:

所有對(duì)象的 gc 元方法是共享的,而不是每次創(chuàng)建 fulluserdata  創(chuàng)建一個(gè)新的元表。這是一個(gè)簡(jiǎn)單的優(yōu)化,可以節(jié)省不少的內(nèi)存。方便起見(jiàn),這個(gè)元表也放在那個(gè)弱表內(nèi)。注意:在 Lua 中,每次壓入一個(gè)  CFucntion 都會(huì)重新分配內(nèi)存創(chuàng)建一個(gè)新對(duì)象。所以應(yīng)該盡可能的共用。

每次從 C 對(duì)象指針生成 fulluserdata 時(shí),都會(huì)去檢查以前是否生成過(guò)。這樣才能使引用計(jì)數(shù)統(tǒng)一計(jì)算。

代碼隨手寫(xiě)的,并沒(méi)有經(jīng)過(guò)嚴(yán)格的測(cè)試,如果誰(shuí)想拿去用可自便,但發(fā)現(xiàn) bug 請(qǐng)通知我修改過(guò)來(lái)。

原文鏈接:http://tech.it168.com/j/2008-02-17/200802171052983.shtml

到此,相信大家對(duì)“怎么在Lua中管理C對(duì)象”有了更深的了解,不妨來(lái)實(shí)際操作一番吧!這里是億速云網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

lua c
AI