lua作为小巧精悍的脚本语言,易于嵌入c/c++中 , 广泛应用于游戏AI ,实际上在任何经常变化的逻辑上都可以使用lua实现,配合c/c++实现的底层接口服务,能够大大降低系统的维护成本。

      lua和c/c++的数据交互通过"栈"进行 ,操作数据时,首先将数据拷贝到"栈"上,然后获取数据,栈中的每个数据通过索引值进行定位,索引值为正时表示相对于栈底的偏移索引,索引值为负时表示相对于栈顶的偏移索引,索引值以1或-1为起始值,因此栈顶索引值永远为-1 ,栈底索引值永远为1 。 "栈"相当于数据在lua和c/c++之间的中转地。每种数据都有相应的存取接口 。

C调用lua中的方法

lua中存储了需要调用的函数

function add(x,y) 	print("sum:"..x+y) 	return x+y end

C语言对其进行调用
#include <lua.hpp> #include <stdio.h> int main() { 	int a=10,b=20; 	int result=0; 	lua_State *L = luaL_newstate(); 	luaopen_base(L); 	luaL_dofile(L,"scene.lua"); 	lua_getglobal(L,"add"); 	lua_pushinteger(L,a); 	lua_pushinteger(L,b); 	lua_pcall(L,2,1,0); 	result=(int)lua_tonumber(L,-1);         lua_close( L); 	printf("result:%d\n",result); } 


  

  

运行结果:

一行一行解释.

1.包含lua的头文件;

2.添加标准输入输出;

5,6.初始化两个变量;

7.创建lua运行环境;

8.加载基本库;

9.加载lua脚本;

10.获取全局变量;

11,12.传入参数到栈;

13.调用函数,不处理结果;

14.结果出栈;

15.关闭lua;

16.打印结果。

C中获取lua中的数据

lua文件用于存储配置文件。

app_name="testApp" id=1 host_info={ hostname="SuperMan", ip="128.0.0.1", } data= { 	date="2013.2", 	user="Jack" } host_info.content=data print("Load .lua successful!") 

  

配置文件中包含字符串,数字还有一个嵌套的table。

下面是用c来解析:

#include <lua.hpp> #include <stdio.h> int table_next(lua_State *L, int i,char **k, char **v) { 	if ( lua_next(L, i) !=0 ) 	{ 		*k = (char *)lua_tostring(L, -2); 		*v = (char *)lua_tostring(L, -1); 		lua_pop(L, 1); 		return 1; 	} 	else 	return 0; } int main() { 	int id=0; 	int ret = 0 ; 	char *k=NULL; 	char *v=NULL; 	lua_State *L = luaL_newstate(); 	luaopen_base(L); 	luaL_dofile(L,"s.lua"); 	lua_pcall(L,0,0,0); 	lua_getglobal(L,"app_name"); 	printf("application name is %s.\n",lua_tostring(L,-1)); 	lua_pop(L,1); 	 	lua_getglobal(L,"host_info"); 	lua_getfield(L,-1,"hostname"); 	lua_getfield(L,-2,"ip"); 	printf("Host name is %s,ip is %s.\n",lua_tostring(L,-2),lua_tostring(L,-1)); 	lua_pop(L,2); 	 	lua_pushnil(L); 	printf("isTable:%d\n",lua_istable(L,-2)); 	while(lua_next(L,-2)) 	{ 		printf("Get key %s\n",lua_tostring(L,-2)); 		if(lua_istable(L,-1)) 		{ 		printf("Is table!\n"); 		lua_getfield(L,-1,"date"); 		lua_getfield(L,-2,"user"); 		printf("\tuser:%s\n",lua_tostring(L,-1)); 		printf("\tdate:%s\n",lua_tostring(L,-2)); 		lua_pop(L,2); 		} 		else 		{ 			printf("value is:%s\n",lua_tostring(L,-1));  		} 		lua_pop(L,1); 	} 	lua_close(L); 	  }


在这里一定要理解交互中的栈模型。

几个重要的函数:

lua_pushnil(lua_State *lua);

pushnil就是向栈中压入Lua的nil,nil在Lua中是一个类型值,在C中,大家可以 将其视为NULL,并且像lua_tostring这样的从栈中未取到值,也就是Lua中的nil,得到的结果就是NULL。

 lua_next(lua_State *lua,int index)

lua_next(lua_State *lua,int index)函数是这个例子的主角,他可以根据指定交互栈中index处的Table,进行遍历,每次取(-1)位置的一个key作为前辈,即将要取得一对元素的上一对元素的key,然后返回Table的该 对元素,将其键先压入栈,再将该键对应的值压入栈,结果就是(-2)位置放的是键,(-1)位置放的是值。Table自然被压入到其后,本例中的(-3)位置。如果key为nil,则默认为首对数据, 会随机的压入一对值。当所有值都被遍历一遍后,next返回0。

lua_pop(lua_State *lua,int num)

该函数如上面所述:从交互句柄的交互栈中弹出num个值。这里不得不说下Lua作配置文件的另一个好处-Lua自己处理堆栈,使得配置文件程序更安全。所有压入栈中的内容,只要 调用该函数,Lua就是自己对其内存进行处理,无需程序员得干预,当然,这样也说明了,不可以带走栈中的内存,也就是不可以将栈中弹出来的内存如字符串内存用作他用,否则可能 在pop后,该内存将失效。


lua_getfield(lua,-1,"name"); 该API主要是用来处理Table。其第一个参数是交互的句柄,第二个参数是Table在交互的栈的位置,第三个参数是前面Table中的键。 该函数的结果是将该Table中对于键的值取出来,并压入到交互栈中。这样就使得原来位于(-1)位置的Table就下压了一个位置到了 (-2)。

关于嵌套table的读取,代码中首先是读取了host_info这个table,当前处于栈的-1的位置,然后pushnil之后,table变为-2的位置,调用next之后,压入一对key-value,可以在-2的位置,value在-1,之前的顺次后移。

知道了位置之后就可以通过相应的get获取值了。


lua调用c库

首先用C写好要注册的函数。

#include <lua.hpp> #include <math.h> #include <stdio.h> typedef int (*lua_CFunction)(lua_State *L); static int l_sin(lua_State *L)  {  double d = luaL_checknumber(L, 1);  lua_pushnumber(L, sin(d));  return 1; /* number of results */ }  static const struct luaL_Reg mylib[]=  {  {"lsin", l_sin},  {NULL, NULL}/* 必须以NULL结尾 */ };  extern int luaopen_mylib (lua_State *L)  {  //lua_register(L,"mylib",mylib);   //luaL_openlib(L, "mylib", mylib, 0);   luaL_register(L,"mylib",mylib);  return 1; }

理论上编译成.so文件即可在lua中调用,但在编译的时候一直报错:

test3.c: In function ‘int luaopen_mytestlib(lua_State*)’: test3.c:44:6: error: cannot convert ‘luaL_Reg*’ to ‘lua_CFunction {aka int (*)(lua_State*)}’ for argument ‘2’ to ‘void lua_pushcclosure(lua_State*, lua_CFunction, int)’ 

至今尚未解决....


lua学习阶段总结

学lua的动机只是因为一句话——合格的程序员应该至少每年学习一门语言!

学得不是很深,大部分的时候都是参照网上的例子敲一些代码,有些需要花时间去理解的地方也没有深究,但至少算是基本掌握了一门脚本语言,以后要用到的话再捡起来也会很快。