佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > Lua教程(二十):Lua調(diào)用C函數(shù)

Lua教程(二十):Lua調(diào)用C函數(shù)

熱門標(biāo)簽:鐵路電話系統(tǒng) 網(wǎng)站排名優(yōu)化 地方門戶網(wǎng)站 Linux服務(wù)器 呼叫中心市場需求 AI電銷 服務(wù)外包 百度競價排名

Lua可以調(diào)用C函數(shù)的能力將極大的提高Lua的可擴展性和可用性。對于有些和操作系統(tǒng)相關(guān)的功能,或者是對效率要求較高的模塊,我們完全可以通過C函數(shù)來實現(xiàn),之后再通過Lua調(diào)用指定的C函數(shù)。對于那些可被Lua調(diào)用的C函數(shù)而言,其接口必須遵循Lua要求的形式,即typedef int (*lua_CFunction)(lua_State* L)。簡單說明一下,該函數(shù)類型僅僅包含一個表示Lua環(huán)境的指針作為其唯一的參數(shù),實現(xiàn)者可以通過該指針進一步獲取Lua代碼中實際傳入的參數(shù)。返回值是整型,表示該C函數(shù)將返回給Lua代碼的返回值數(shù)量,如果沒有返回值,則return 0即可。需要說明的是,C函數(shù)無法直接將真正的返回值返回給Lua代碼,而是通過虛擬棧來傳遞Lua代碼和C函數(shù)之間的調(diào)用參數(shù)和返回值的。這里我們將介紹兩種Lua調(diào)用C函數(shù)的規(guī)則。

1. C函數(shù)作為應(yīng)用程序的一部分。

復(fù)制代碼 代碼如下:

#include stdio.h>
#include string.h>
#include lua.hpp>
#include lauxlib.h>
#include lualib.h>

//待Lua調(diào)用的C注冊函數(shù)。
static int add2(lua_State* L)
{
    //檢查棧中的參數(shù)是否合法,1表示Lua調(diào)用時的第一個參數(shù)(從左到右),依此類推。
    //如果Lua代碼在調(diào)用時傳遞的參數(shù)不為number,該函數(shù)將報錯并終止程序的執(zhí)行。
    double op1 = luaL_checknumber(L,1);
    double op2 = luaL_checknumber(L,2);
    //將函數(shù)的結(jié)果壓入棧中。如果有多個返回值,可以在這里多次壓入棧中。
    lua_pushnumber(L,op1 + op2);
    //返回值用于提示該C函數(shù)的返回值數(shù)量,即壓入棧中的返回值數(shù)量。
    return 1;
}

//另一個待Lua調(diào)用的C注冊函數(shù)。
static int sub2(lua_State* L)
{
    double op1 = luaL_checknumber(L,1);
    double op2 = luaL_checknumber(L,2);
    lua_pushnumber(L,op1 - op2);
    return 1;
}

const char* testfunc = "print(add2(1.0,2.0)) print(sub2(20.1,19))";

int main()
{
    lua_State* L = luaL_newstate();
    luaL_openlibs(L);
    //將指定的函數(shù)注冊為Lua的全局函數(shù)變量,其中第一個字符串參數(shù)為Lua代碼
    //在調(diào)用C函數(shù)時使用的全局函數(shù)名,第二個參數(shù)為實際C函數(shù)的指針。
    lua_register(L, "add2", add2);
    lua_register(L, "sub2", sub2);
    //在注冊完所有的C函數(shù)之后,即可在Lua的代碼塊中使用這些已經(jīng)注冊的C函數(shù)了。
    if (luaL_dostring(L,testfunc))
        printf("Failed to invoke.\n");
    lua_close(L);
    return 0;
}

2. C函數(shù)庫成為Lua的模塊。

將包含C函數(shù)的代碼生成庫文件,如Linux的so,或Windows的DLL,同時拷貝到Lua代碼所在的當(dāng)前目錄,或者是LUA_CPATH環(huán)境變量所指向的目錄,以便于Lua解析器可以正確定位到他們。在我當(dāng)前的Windows系統(tǒng)中,我將其copy到"C:\Program Files\Lua\5.1\clibs\",這里包含了所有Lua可調(diào)用的C庫。見如下C語言代碼和關(guān)鍵性注釋:

復(fù)制代碼 代碼如下:

#include stdio.h>
#include string.h>
#include lua.hpp>
#include lauxlib.h>
#include lualib.h>

//待注冊的C函數(shù),該函數(shù)的聲明形式在上面的例子中已經(jīng)給出。
//需要說明的是,該函數(shù)必須以C的形式被導(dǎo)出,因此extern "C"是必須的。
//函數(shù)代碼和上例相同,這里不再贅述。
extern "C" int add(lua_State* L)
{
    double op1 = luaL_checknumber(L,1);
    double op2 = luaL_checknumber(L,2);
    lua_pushnumber(L,op1 + op2);
    return 1;
}

extern "C" int sub(lua_State* L)
{
    double op1 = luaL_checknumber(L,1);
    double op2 = luaL_checknumber(L,2);
    lua_pushnumber(L,op1 - op2);
    return 1;
}

//luaL_Reg結(jié)構(gòu)體的第一個字段為字符串,在注冊時用于通知Lua該函數(shù)的名字。
//第一個字段為C函數(shù)指針。
//結(jié)構(gòu)體數(shù)組中的最后一個元素的兩個字段均為NULL,用于提示Lua注冊函數(shù)已經(jīng)到達(dá)數(shù)組的末尾。
static luaL_Reg mylibs[] = {
    {"add", add},
    {"sub", sub},
    {NULL, NULL}
};

//該C庫的唯一入口函數(shù)。其函數(shù)簽名等同于上面的注冊函數(shù)。見如下幾點說明:
//1. 我們可以將該函數(shù)簡單的理解為模塊的工廠函數(shù)。
//2. 其函數(shù)名必須為luaopen_xxx,其中xxx表示library名稱。Lua代碼require "xxx"需要與之對應(yīng)。
//3. 在luaL_register的調(diào)用中,其第一個字符串參數(shù)為模塊名"xxx",第二個參數(shù)為待注冊函數(shù)的數(shù)組。
//4. 需要強調(diào)的是,所有需要用到"xxx"的代碼,不論C還是Lua,都必須保持一致,這是Lua的約定,
//   否則將無法調(diào)用。
extern "C" __declspec(dllexport)
int luaopen_mytestlib(lua_State* L)
{
    const char* libName = "mytestlib";
    luaL_register(L,libName,mylibs);
    return 1;
}

見如下Lua代碼:

復(fù)制代碼 代碼如下:

 require "mytestlib"  --指定包名稱
 
 --在調(diào)用時,必須是package.function
 print(mytestlib.add(1.0,2.0))
 print(mytestlib.sub(20.1,19))

您可能感興趣的文章:
  • Lua教程(四):在Lua中調(diào)用C語言、C++的函數(shù)
  • Lua教程(三):C語言、C++中調(diào)用Lua的Table示例
  • C++中調(diào)用Lua函數(shù)實例
  • Lua中調(diào)用C++函數(shù)實例
  • Lua調(diào)用自定義C模塊
  • Lua中調(diào)用C語言函數(shù)實例
  • Lua教程(十九):C調(diào)用Lua
  • Lua編程示例(六): C語言調(diào)用Lua函數(shù)
  • C語言中調(diào)用Lua函數(shù)實例
  • C語言與Lua之間的相互調(diào)用詳解

標(biāo)簽:蘭州 衡水 崇左 湘潭 仙桃 湖南 黃山 銅川

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Lua教程(二十):Lua調(diào)用C函數(shù)》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    昭苏县| 石楼县| 子长县| 邹平县| 安多县| 若羌县| 昂仁县| 鹤庆县| 广水市| 洮南市| 疏勒县| 黑河市| 咸宁市| 蒲城县| 玉树县| 屏东县| 偏关县| 福鼎市| 宁德市| 浙江省| 德兴市| 宁陵县| 晋宁县| 晋江市| 南通市| 抚松县| 鄂温| 梁河县| 清涧县| 马关县| 筠连县| 寻乌县| 思南县| 全南县| 五原县| 永仁县| 乾安县| 鹿邑县| 天柱县| 定日县| 焉耆|