­

Lua C++交互 應用實例步驟(UserData使用)

一.配置Lua C++交互環境

    1.下載Lua 包環境

  地址: //www.lua.org/download.html ,我們這裡用的是5.4.2版本。

  2.新建C++ 控制台應用程序

  3.導入Lua 源碼

  1)導入Lua 源碼到同級目錄(這樣做為了Main函數所在類添加Lua 的C 類的時候直接可以include,不需要考慮目錄層級問題)

  

 

   打開的目錄里,將5.4.2 的lua包里的 src目錄里的文件全部拷到此目錄

  2)添加Lua 源碼

  

 

   選中剛剛導入的Lua文件

  3)編譯

  這時,點擊目錄生成,回發現報錯

 

 這是關鍵報錯信息,重複main頭文件,眾所周知,main函數是啟動函數;那就注釋調main.

會發現,有兩處main ,lua.c 中 int main 和luac.c int main,分別都注釋了。(或者簡單粗暴,直接將luc.c和luac.c直接刪除也可以)

再點生成,成功。這兩處main,其實就是如果你將此lua編碼打包成exe,就是啟動函數。但是我們這裡不需要這邊的main,我們自己新建的C++ 應用程序,是有自己的Main。

二.Hello.lua 運行

  新建Lua腳本,必須.lua 結尾。

 

  1.關鍵API

  1)lua與C++交互,必須要一個虛擬棧。可通過lua_State* L = luaL_newstate(); 對於棧的介紹可參考://blog.csdn.net/shun_fzll/article/details/39120965,這裡就不對棧作描述了。

  2)加載Lua文件,可通過 luaL_dofile 返回一個 int類型,返回結果不為0,則表示異常

  3)Lua有異常,會把返回結果返回到棧頂,也就是說報異常了,可以通過lua_tostring(L,-1)得到報啥錯,再打印出來.

  4)lua_close 就是釋放這個棧。

  5)對於C++調用 C的頭文件,需要extern

 2.實例分析

 1 #include <iostream>
 2 #include <string.h>
 3 using namespace std;
 4 extern "C"
 5 {
 6 #include "lua.h"
 7 #include "lualib.h"
 8 #include "lauxlib.h"
 9 }
10 int main()
11 {
12     std::cout << "Hello World!\n";
13     lua_State* L = luaL_newstate();
14     int Ret = luaL_dofile(L, "LuaHello.lua");
15     if (Ret)
16     {
17         string strError = lua_tostring(L, -1);
18         cout << strError.c_str() << endl;
19         return -1;
20     }
21         
22     lua_close(L);
23     return 0;
24 }

 

  1)運行上述代碼

 

   2)想一下,好像lua代碼是運行了,但是print怎麼會nil,並報錯,提示的還挺智能,具體行號都報出來了。

我們再試着修改Lua代碼中下移兩行試試:

 

  3)咦,那就證明Lua文件是肯定運行了,就是異常了。找不到print,print不是lua庫里的函數嘛,為啥拿不到了。

因為C++環境中創建Lua_State這個棧,這個默認並沒有打開lua標準庫。所以查一下。

在 上述Line:13 後面新增 luaL_openlibs(L);

  運行 得到 

 

  恭喜,C++ 與Lua通信成功。

三.Lua UserData應用實例

  關於具體lua與C交互的一些常見操作,可參考://blog.csdn.net/shun_fzll/article/details/39120965

  

  問題:有時候,你會想,我在C++中聲明了一個struct,在lua中,也想聲明這個struct,怎麼弄納?接下來這個案例就是教你如何使。

我們的UserData就是解決這個問題的。

  Lua UserData分兩種,我們這說的是FullUserData,就是可以解決上述問題的UserData.

  直奔主題,看代碼分析。

  Lua5.1之前和之後的註冊函數差距很大,詳情可參考://blog.csdn.net/ljhjason/article/details/28860633

  Lua5.1之後可以使用 luaL_requiref 代替 Lua5.1之前的luaL_register ,可參考://blog.csdn.net/ljhjason/article/details/28860633

  

 1 #include <iostream>
 2 #include <string.h>
 3 using namespace std;
 4 extern "C"
 5 {
 6 #include "lua.h"
 7 #include "lualib.h"
 8 #include "lauxlib.h"
 9 }
10 
11 struct TestCStruct
12 {
13     char* Name;
14     int Num = 1;
15 };
16 
17 int NewTestCStruct(lua_State* L)
18 {
19     size_t iBytes = sizeof(struct TestCStruct);
20     struct TestCStruct* pStruct = (struct TestCStruct*)lua_newuserdata(L, iBytes);
21     return 1;
22 }
23 
24 int SetNum(lua_State* L)
25 {
26     struct TestCStruct *pStruct = (struct TestCStruct*)lua_touserdata(L, 1);
27     luaL_argcheck(L, pStruct != NULL, 1, "Error Param");
28     int num = luaL_checkinteger(L, 2);
29     luaL_argcheck(L, num > 0, 2,"Wrong Paramter");
30     pStruct->Num = num;
31     return 1;
32 }
33 
34 int GetNum(lua_State* L)
35 {
36     struct TestCStruct *pStruct = (struct TestCStruct*)lua_touserdata(L, 1);
37     luaL_argcheck(L, pStruct != NULL, 1, "ErrorStruct");
38     lua_pushinteger(L, pStruct->Num);
39     return 1;
40 }
41 
42 struct luaL_Reg arrayFunc[] = {
43 {"new",NewTestCStruct},
44 {"SetNum",SetNum},
45 {"SNum",GetNum},
46 {NULL,NULL}
47 };
48 
49 LUALIB_API int luaopen_mytest(lua_State* L)
50 {
51     luaL_newlib(L, arrayFunc);
52     return 1;
53 }
54 
55 int main()
56 {
57     std::cout << "Hello World!\n";
58     lua_State* L = luaL_newstate();
59     luaL_openlibs(L);
60     luaL_requiref(L, "mytest", luaopen_mytest, 0);
61     int Ret = luaL_dofile(L, "LuaHello.lua");
62     if (Ret)
63     {
64         string strError = lua_tostring(L, -1);
65         cout << strError.c_str() << endl;
66         return -1;
67     }
68 
69     lua_close(L);
70     return 0;
71 }

Lua中源碼

local t = require("mytest")
print("t:" .. tostring(t))

local y = t.new()
print("y :" .. tostring(y))
t.SetNum(y,100)
print("Num:" ..t.SNum(y))
local y2 = t.new()
print("y2 :" .. tostring(y2))
print("Hello lua and C++")

運行結果

 綜上, 可以看到Full UserData 的使用,主要核心是Line:60 luaL_requiref的使用。

//files.cnblogs.com/files/u3ddjw/TestLuaAndC.rar