亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > 綜合 > 正文

把Lua函數傳遞到C/C++中實例

2024-07-21 23:04:17
字體:
來源:轉載
供稿:網友

問題

在Lua中,因為函數也是第一類值,所以會出現將函數作為另一個函數的參數,或者函數作 為函數的返回值。這種機制在很多地方都能代碼更靈活更簡潔,例如:

 

復制代碼 代碼如下:

table.sort(table [,comp])

 

這里的comp就要求傳入一個函數,我們在調用時,大概會有如下形式:

 

復制代碼 代碼如下:

table.sort(t, comp) -- 直接寫函數名
    table.sort(t, local_comp) -- 某個局部函數
    table.sort(t, function (a, b) xxx end ) -- 臨時構造一個匿名函數

 

其中最后一種方式最為靈活,任意時候在需要的時候構造一個匿名函數。這種在Lua自身的 環境中使用,自然沒有問題。但是,當我們在C/C++中注冊一些函數到Lua環境中,而這些 函數也需要使用函數參數的時候,問題就出來了。

Lua本身是不支持將Lua函數作為函數參數傳入C/C++的,不管這個想要傳入的函數是全局的 、局部的、或者匿名的(匿名的本質上也算局部的)。一般情況下,我們唯一的交互方式, 不是傳入一個函數,而是一個全局函數名。C/C++保存這個函數名,在需要回調Lua的時候, 就在Lua全局表中找到這個函數(根據函數名),然后再調用之。情況大致如下:

 

復制代碼 代碼如下:

function lua_func () xxx end
    cfunc(lua_func) -- wrong
    cfunc("lua_func") -- right

 

我們這回的腳本模塊,策劃會大量使用需要回調函數的C/C++函數。顯然,創建大量的全局 函數,先是從寫代碼的角度看,就是很傷神的。

解決

我們最終需要的方式,大概如下:

 

復制代碼 代碼如下:

cfunc(lua_func) -- ok
    cfunc(function () xxx end) -- ok
    local xxx = function () xxx end
    cfunc(xxx) -- ok

 

要解決這個問題,我的思路是直接在Lua層做一些包裝。因為C/C++那邊僅支持傳入一個全局 函數名(當然不一定得全局的,根據實際情況,可能在其他自己構造的表里也行),也就是 一個字符串,所以我的思路就是將Lua函數和一個唯一的字符串做映射。

 

復制代碼 代碼如下:

function wrap (fn)
        local id = generate_id()
        local fn_s = "__callback_fn"..id
        _G[fn_s] = fn
        return fn_s
    end

 

這個wrap函數,就是將一個函數在全局表里映射到一個字符串上,那么在使用時:

 

復制代碼 代碼如下:

cfunc(wrap(function () xxx end))
    cfunc(const char *fn_name, xxx); -- cfunc的原型

 

cfunc是C/C++方注冊進Lua的函數,它的原型很中規中矩,即:只接收一個函數名,一個字 符串,如之前所說,C/C++要調用這個回調函數時,就根據這個字符串去查找對應的函數。 腳本方在調用時,如果想傳入一個匿名函數了,就調用wrap函數包裝一下即可。

一個改進

上面的方法有個很嚴重的問題,在多次調用wrap函數后,將導致全局表也隨之膨脹。我們需 要想辦法在C/C++完成回調后,來清除wrap建立的數據。這個工作當然可以放到C/C++來進行 ,例如每次發生回調后,就設置下全局表。但這明顯是不對的,因為違背了接口的設計原則 ,這個額外的機制是在Lua里添加的,那么責任也最好由Lua來負。要解決這個問題,就可以 使用Lua的metamethods機制。這個機制可以在Lua內部發生特定事件時,讓應用層得到通知。 這里,我們需要關注__call事件。

Lua中只要有__call metamethod的值,均可被當作函數調用。例如:

 

復制代碼 代碼如下:

ab(1, 2)

 

這里這個函數調用形式,Lua就會去找ab是否有__call metamethod,如果有,則調用它。這 個事實暗示我們,一個table也可以被調用。一個改進的wrap函數如下:

 

復制代碼 代碼如下:

local function create_callback_table (fn, name)
        local t = {}
        t.callback = fn
        setmetatable (t, {__call =  -- 關注__call
            function (func, ...) -- 在t(xx)時,將調用到這個函數
                func.callback (...) -- 真正的回調
                del_callback (name) -- 回調完畢,清除wrap建立的數據
            end })
        return t
    end
   
    function wrap (fn)
        local id = generate_func_id() -- 產生唯一的id
        local fn_s = "_callback_fn"..id
        _G[fn_s] = create_callback_table(fn, fn_s) -- _G[fn_s]對應的是一個表
        return fn_s
    end

 

在我們的C/C++程序中,依然如往常一樣,先是從_G里取出函數名對應的對象。雖然這個對 象現在已經是一個table。然后lua_call。

上面的代碼是否會在原有基礎上增加不可接受的性能代價?雖然我沒有做過實際測試,但是 從表明看來,排除meta table在Lua里的代價,也就多了幾次Lua函數調用。

最后,感嘆一下,Lua里的table及metatable機制,實在非常強大。這種強大不是功能堆砌 出來的強大,而是簡單東西組合出來的強大。其背后的設計思想,著實讓人佩服。

4.26.2011 Update

之前的文中說“Lua本身是不支持將Lua函數作為函數參數傳入C/C++的“,這句話嚴格來說不 正確(由某網友評論)。假設函數cfun由c/c++注冊,我們是可以編寫如下代碼的:

 

復制代碼 代碼如下:

cfunc(print) -- 傳入Lua函數

 

但是問題在于,我們無法取出這個函數并保存在c/c++方。Lua提供了一些接口用于取cfunc 的參數,例如luaL_checknumber(封裝lua_tonumber)。但沒有類似luaL_checkfunction的 接口。Lua中的table有同樣的問題。究其原因,主要是Lua中的函數沒有直接的c/c++數據結 構對應。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产成人精品一区二区| 91精品国产91久久久久久最新| 久久人人爽人人爽人人片av高请| 国产精品影片在线观看| 国产精品91免费在线| 国产一区二区三区在线| 国产一区二区色| 亚洲视频电影图片偷拍一区| 国产精品福利观看| 国产福利视频一区| 国产精品久久色| 欧美一区二区三区免费视| 欧美激情视频一区二区| 亚洲综合大片69999| 日韩欧美国产骚| 国产精品日韩欧美| 欧美资源在线观看| 永久免费精品影视网站| 久久九九精品99国产精品| 成人有码在线视频| 搡老女人一区二区三区视频tv| 国产97在线|亚洲| 亚洲欧美在线磁力| 最好看的2019的中文字幕视频| 久久伊人91精品综合网站| 国产成人精品久久久| wwwwwwww亚洲| 国内精品一区二区三区| 国产成+人+综合+亚洲欧洲| 久久精品99久久久久久久久| 欧美肥老太性生活视频| 国产精品久久久999| 国产精品黄色av| 久久精品成人动漫| 91精品国产91久久久久久最新| 亚洲欧美日韩第一区| 国产亚洲精品久久久久久| 国产精品海角社区在线观看| 91免费综合在线| 精品国产成人在线| 久久久国产精彩视频美女艺术照福利| 亚洲性生活视频| 亚洲国产精品久久久久秋霞不卡| 久久久国产视频| 成人国产精品av| 日本最新高清不卡中文字幕| 久久免费视频在线| 中文字幕日韩在线视频| 伊人久久综合97精品| 亚洲成人黄色网| 国产精品三级网站| 中文字幕少妇一区二区三区| 久久精品视频免费播放| 91亚洲精品视频| 不卡av电影院| 国产亚洲视频在线| 国产精品久久久久久久久影视| 欧美精品在线免费| 一区二区日韩精品| 精品在线观看国产| 最近2019免费中文字幕视频三| 亚洲色图偷窥自拍| 精品中文视频在线| 国产精品人人做人人爽| 久色乳综合思思在线视频| 亚洲欧美国产日韩天堂区| 日本一区二区在线免费播放| 亚洲欧美成人一区二区在线电影| 国产mv久久久| 亚洲欧洲一区二区三区在线观看| 亚洲高清福利视频| 欧美尺度大的性做爰视频| 日本中文字幕久久看| 一夜七次郎国产精品亚洲| 97视频色精品| 日韩免费av一区二区| 久久躁狠狠躁夜夜爽| 午夜精品一区二区三区在线播放| 1769国产精品| 欧美精品电影免费在线观看| 日韩经典一区二区三区| 91精品国产高清久久久久久久久| 欧美第一黄网免费网站| 久久99久久99精品中文字幕| 亚洲护士老师的毛茸茸最新章节| 亚洲美女av在线播放| 在线看福利67194| 亚洲男人天堂2023| 精品国产福利视频| 久久精品国产免费观看| 上原亚衣av一区二区三区| 久久久久久97| 亚洲第一视频网站| 欧美在线一区二区三区四| 久久黄色av网站| 久久久久久久久久久91| 欧美日韩亚洲91| 国产精品免费在线免费| 午夜精品久久久久久久99热| 狠狠色狠狠色综合日日小说| 欧美一级视频免费在线观看| 国产深夜精品福利| 欧美巨乳美女视频| 国产日韩欧美视频| 精品久久久久久久久久国产| 久色乳综合思思在线视频| 欧美国产视频日韩| 久久久电影免费观看完整版| 一区二区三区四区在线观看视频| 欧美日韩国产第一页| 亚洲精品天天看| 欧美最猛性xxxxx免费| 神马久久久久久| 日韩资源在线观看| 狠狠躁天天躁日日躁欧美| 久久99视频免费| 538国产精品一区二区在线| 国产精品xxxxx| 按摩亚洲人久久| 亚洲一区二区自拍| 亚洲人永久免费| 国产精品女人网站| 久久噜噜噜精品国产亚洲综合| 亚洲开心激情网| 91九色综合久久| 国产精品美女无圣光视频| 久久中文精品视频| 色小说视频一区| 中文字幕在线看视频国产欧美| 亚洲精品久久7777777| 国产精品国产福利国产秒拍| 午夜剧场成人观在线视频免费观看| 麻豆一区二区在线观看| 伊人青青综合网站| 亚洲色图狂野欧美| 日本精品中文字幕| 日韩免费中文字幕| 午夜精品福利视频| 国产精品嫩草影院久久久| 欧美视频在线观看免费网址| 日韩精品一二三四区| 中文字幕一区二区三区电影| 国产网站欧美日韩免费精品在线观看| 美日韩丰满少妇在线观看| 91av福利视频| 91色视频在线导航| 91精品国产成人| 色综合久综合久久综合久鬼88| 久久久久久免费精品| 91精品免费久久久久久久久| 色综合导航网站| 日韩av电影免费观看高清| 久久手机免费视频| 狠狠综合久久av一区二区小说| 久久99国产精品久久久久久久久| 日韩日本欧美亚洲| 久久人91精品久久久久久不卡| 国产91对白在线播放| 亚洲无限乱码一二三四麻| 久久精品国产清自在天天线| 成人黄色av播放免费| 亚洲第一区中文99精品| 精品国产网站地址| 久久久精品久久久|