zoukankan      html  css  js  c++  java
  • lua与c的交互(函数专用)

    Lua与C的交互

    Lua是一个嵌入式的语言,它不仅可以是一个独立运行的程序,也可以是一个用来嵌入其它应用的程序库。

    C API是一个C代码与Lua进行交互的函数集,它由以下几部分构成:

    1、  读写Lua全局变量的函数;

    2、  调用Lua函数的函数;

    3、  运行Lua代码片段的函数;

    4、  注册C函数后可以在Lua中被调用的函数;

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

    另外,还可以使用栈来保存临时变量。栈的使用解决了C和LUA之间两个不协调的问题:

    1、  Lua使用自动垃圾收集机制,而C要求显式的分配和释放内存;

    2、  Lua使用动态数据类型,而C使用静态类型;

    特别注意的是:

    1、每当Lua调用C函数时,C函数会使用一个局部栈,这个局部栈与之前的栈,以及其它正在调用的C函数使用的栈都是相互独立的。Lua和C就使用这个局部的栈进行数据交互。

    2、当Lua调用C时,栈至少包含LUA_MINSTACK(20)个位置,程序员也可以使用lua_checkstack函数来增加栈的大小。

    3、使用伪索引(Pseudo-Indices)来表示一些不在栈中的数据,比如thread环境、C函数环境、registry、C闭包的upvalues。

        thread环境(全局变量也在这里),使用伪索引 LUA_GLOBALSINDEX;

        运行中的C函数环境,使用伪索引 LUA_ENVIRONINDEX;

        Registry,使用伪索引 LUA_REGISTRYINDEX;

        C闭包的upvalues,可以使用lua_upvalueindex(n)来访问第n个upvalue;

    关于Registry:

    在C里面如果函数要保存持久状态,只能依靠全局或static变量,但这样C API库就无法为多个LuaState状态同时提供服务(就类似于带有static变量的C函数是不可重入的)。为此Lua提供了一个名为Registry的预定义table,允许C API往Registry里面存储数据。

    关于References:

    Reference其实就是在一个指定的表t中保存一次lua的数据对象,Refenence本身其实就是表t的索引子,简称RefIndex,当RefIndex作为Refenence时,t

    [RefIndex]其实就是用户要求引用的lua数据对象。当RefIndex被luaL_unref()回收时,t每一个被回收的RefIndex构成一个单向链表: t[Refindex] = Refindex0,

    t[Refindex0] = Refindex1, t[Refindex1] = Refindex2 ... t[0] = Refindex。

    注意,t[0]始终是指向了空闲链表的头部。每次调用luaL_ref()且回收链表为空时,都会产生一个新的Reference,每次调用luaL_unref()都会销毁一个指定的

    Reference存入空闲链表中。

    C API接口

    类型声明

    typedef double lua_Number;

    typedef ptrdiff_t lua_Integer;

    初始化lua状态机

    lua_State* lua_open();

    lua_State *lua_newstate (lua_Alloc f, void *ud);

    lua_newstate 创建一个新的、独立的Lua状态机,如果因为内存不足导致创建失败,返回NULL。参数f 指定内存分配函数,参数ud是传给f 函数的指针。

    lua_open 没有指定内存分配函数的功能,不建议再使用。

    注意:lua_State表示的一个Lua程序的执行状态,它代表一个新的线程(注意是指Lua中的thread类型,不是指操作系统中的线程),每个thread拥有独立的数据

    栈以及函数调用链,还有独立的调试钩子和错误处理方法。

    销毁lua状态机

    void lua_close(lua_State *L);

    销毁Lua状态机的所有对象,回收分配的内存。

    加载lua库

    void luaL_openlibs(lua_State *L);                  

    void luaopen_base(lua_State *L);

    void luaopen_package(lua_State *L);

    void luaopen_string(lua_State *L);

    void luaopen_io(lua_State *L);

    void luaopen_table(lua_State *L);

    void luaopen_math(lua_State *L);

    luaL_openlibs 在给定的Lua状态机中打开所有的标准Lua库;

    编译/加载 lua代码

    int luaL_dofile(lua_State *L, char *lua_script);

    int luaL_dostring (lua_State *L, const char *str);

    int lua_load (lua_State *L, lua_Reader reader, void *data,const char *chunkname);

    int luaL_loadbuffer (lua_State *L, const char *buff, size_t sz, const char *name);

    int luaL_loadfile (lua_State *L, const char *filename);

    int luaL_loadstring (lua_State *L, const char *s);

    luaL_dofile 加载并执行给定lua文件,成功返回0,错误返回1;

    luaL_dostring 加载并执行给定string,成功返回0,错误返回1;

    lua_load 加载一段chunk(但并不执行它),并将编译后的代码作为一个函数压入堆栈,如果发生错误,则将错误消息压入堆栈;

    luaL_loadbuffer 从一个buffer中加载chunk;

    luaL_loadfile从文件加载chunk;

    luaL_loadstring从字符串加载chunk;

    函数参数检查

    void luaL_checkany (lua_State *L, int narg);

    int luaL_checkint (lua_State *L, int narg);

    lua_Integer luaL_checkinteger (lua_State *L, int narg);

    long luaL_checklong (lua_State *L, int narg);

    const char *luaL_checklstring (lua_State *L, int narg, size_t *l);

    lua_Number luaL_checknumber (lua_State *L, int narg);

    int luaL_checkoption (lua_State *L, int narg, const char *def, const char *const lst[]);

    const char *luaL_checkstring (lua_State *L, int narg);

    void luaL_checktype (lua_State *L, int narg, int t);

    void *luaL_checkudata (lua_State *L, int narg, const char *tname);

    luaL_checkany 检查函数有多少个(由narg指定)参数;

    luaL_checktype 检查函数第narg个参数是否为指定类型;

    luaL_checkudata 检查函数第narg个参数是否为name类型的userdata;

    luaL_checkint/luaL_checkinterger 等接口非常类似:

    以luaL_checkint为例,它是检查函数的第narg个参数类型是否number型,并且返回这个number型参数;

    luaL_checkoption 检查函数第narg个参数是否位字符串类型,并且在lst[](字符串数组)中搜索这个字符串,最后返回匹配的数组下标,如未能匹配,引发一个

    错误。如果参数def非空,当narg参数不存在或为nil时,就是要def作为搜索串。这个函数的作用是将字符串映射为C的enum类型。

    table操作

    void lua_createtable (lua_State *L, int narr, int nrec);

    void lua_newtable (lua_State *L);

    void lua_settable (lua_State *L, int index);

    void lua_gettable (lua_State *L, int index);

    lua_createtable 创建一个空table并压入堆栈,它会为narr个数组风格元素,和nrec个记录风格元素预分配内存空间。

    lua_newtable 创建一个空table,并压入stack,等价于 lua_createtable(L, 0, 0);

    lua_settable 相当于t[k]=v 操作,其中值t由参数index指定,v是栈顶,k是栈顶下一个元素;这个函数会将key和value都弹出栈;

    lua_gettable 将t[k]压入堆栈,由参数index指定操作的table,k是栈顶元素。这个函数会弹出栈顶的key,并由t[k]代替;

    metatable操作

    int luaL_newmetatable (lua_State *L, const char *tname);

    void luaL_getmetatable (lua_State *L, const char *tname);

    int lua_getmetatable (lua_State *L, int index);

    int lua_setmetatable (lua_State *L, int index);

    int luaL_getmetafield (lua_State *L, int obj, const char *e);

    luaL_newmetatable 在Registry中创建一个key为tname的metatable,并返回1;如果Registry 已经有tname这个key,则返回0;这两种情况都会将metatable压入

    堆栈;

    luaL_getmetatable 将Registry中key为tname的metatable压入堆栈;

    luaL_getmetatable 将index处的元表压入堆栈;

    lua_setmetatable 弹出栈顶,并将它作为由index指定元素的元表;

    luaL_getmetafield 将索引obj处的元素的元表的e字段压入堆栈;

    stack操作

    void lua_pushboolean (lua_State *L, int b);

    void lua_pushcclosure (lua_State *L, lua_CFunction fn, int n);

    void lua_pushcfunction (lua_State *L, lua_CFunction f);

    const char *lua_pushfstring (lua_State *L, const char *fmt, ...);

    void lua_pushinteger (lua_State *L, lua_Integer n);

    void lua_pushlightuserdata (lua_State *L, void *p);

    void lua_pushliteral (lua_State *L, const char *s);

    void lua_pushlstring (lua_State *L, const char *s, size_t len);

    void lua_pushnil (lua_State *L);

    void lua_pushstring (lua_State *L, const char *s);

    lua_pushfstring 将一个格式化字符串压入堆栈,并返回指向这个字符串的指针;

    lua_pushlstring 将一个指定大小的字符串压入堆栈;

    lua_pushvalue 将栈中指定索引的元素复制一份到栈顶;

    值得注意的是,向栈中压入一个元素时,应该确保栈中具有足够的空间,可以调用lua_checkstack来检测是否有足够的空间。

    实质上这些API是把C语言里面的值封装成Lua类型的值压入栈中的,对于那些需要垃圾回收的元素(如string、full userdata),在压入栈时,都会在Lua(也就是Lua虚拟机中)生成一个副本,从此不会再依赖于原来的C值。例如lua_pushstring 向栈中压入一个以''结尾的字符串,在C中调用这个函数后,可以任意修改或释放这个字符串,也不会出现问题。

    void lua_pop(lua_State *L, int n);

    int lua_gettop (lua_State *L);

    void lua_concat (lua_State *L, int n);

    void lua_getfield (lua_State *L, int index, const char *k);

    void lua_setfield (lua_State *L, int index, const char *k);

    void lua_getglobal(lua_State *L, char *name);

    void lua_setglobal (lua_State *L, const char *name);

    void lua_insert (lua_State *L, int index);

    void lua_remove (lua_State *L, int index);

    void lua_replace (lua_State *L, int index);

    int lua_next (lua_State *L, int index);

    size_t lua_objlen (lua_State *L, int index);

    void luaL_checkstack (lua_State *L, int sz, const char *msg);

     lua_pop 从栈中弹出n个元素;

    lua_gettop 返回栈顶元素的索引(也即元素个数);

    lua_concat 将栈顶开始的n个元素连接起来,并将它们出栈,然后将结果入栈;

    lua_getfield 将t[k]压入堆栈,t由参数index指定在栈中的位置;

    lua_setfield 相当于t[k]=v,t由参数index指定在栈中的位置,v是栈顶元素,改函数会将栈顶的value出栈;

    lua_getglobal(L,s) 等价于 lua_getfield(L, LUA_GLOBALSINDEX, s),注意:栈中LUA_GLOBALSINDEX索引位置处是当前Lua状态机的全局变量环境。

    lua_setglobal(L,s) 等价于 lua_setfield(L, LUA_GLOBALSINDEX, s);

    lua_insert 移动栈顶元素到index指定的位置;

    lua_remove 移除index处的元素,index之上的元素均下移一个位置;

    lua_replace 将栈顶元素移到指定位置,并取代原来的元素,原先的栈顶元素弹出;

    lua_next 弹出一个key,然后将t[key]入栈,t是参数index处的table;在利用lua_next遍历栈中的table时,对key使用lua_tolstring尤其需要注意,除非知道

    key都是string类型。

    lua_objlen 返回index处元素的长度,对string,返回字符串长度;对table,返回"#"运算符的结果;对userdata,返回内存大小;其它类型返回0;

    luaL_checkstack 增加栈大小(新增sz个元素的空间),如果grow失败,引发一个错误,msg参数传递错误消息。

    int lua_isboolean (lua_State *L, int index);

    int lua_iscfunction (lua_State *L, int index);

    int lua_isfunction (lua_State *L, int index);

    int lua_islightuserdata (lua_State *L, int index);

    int lua_isnil (lua_State *L, int index);

    int lua_isnone (lua_State *L, int index);

    int lua_isnoneornil (lua_State *L, int index);

    int lua_isnumber (lua_State *L, int index);

    int lua_isstring (lua_State *L, int index);

    int lua_istable (lua_State *L, int index);

    int lua_isthread (lua_State *L, int index);

    int lua_isuserdata (lua_State *L, int index);

    这些都是栈中指定元素类型检查的接口;

    函数调用

    void lua_call (lua_State *L, int nargs, int nresults);

    int lua_pcall (lua_State *L, int nargs, int nresults, int errfunc);

    int lua_cpcall (lua_State *L, lua_CFunction func, void *ud);

    lua_call 调用函数,参数nargs指定函数参数个数,参数nresults指定返回值个数。首先,被调函数必须在栈中;其次,函数参数必须是按从左往右的顺序入栈的;函数调用时,所有函数参数都会弹出堆栈。函数返回时,其返回值入栈(第一个返回最最先入栈)。

    lua_pcall 以保护模式调用函数,如果发生错误,捕捉它,并将错误消息压入栈,然后返回错误码。

    lua_cpcall 以保护模式调用C函数func,参数ud指针指向一个用户自定义数据。

    错误处理

    int luaL_error (lua_State *L, const char *fmt, ...);

    引发一个错误。

    类型转换

    int lua_toboolean (lua_State *L, int index);

    lua_CFunction lua_tocfunction (lua_State *L, int index);

    lua_Integer lua_tointeger (lua_State *L, int index);

    const char *lua_tolstring (lua_State *L, int index, size_t *len);

    lua_Number lua_tonumber (lua_State *L, int index);

    const void *lua_topointer (lua_State *L, int index);

    const char *lua_tostring (lua_State *L, int index);

    lua_State *lua_tothread (lua_State *L, int index);

    void *lua_touserdata (lua_State *L, int index);

    lua_toboolean 将给定index索引处的元素转换为bool类型(0或1);

    lua_tocfunction 将给定index索引处的元素转换为C函数;

    lua_tointeger 将给定index索引处的元素转换为int类型;

    lua_tolstring 将给定index索引处的元素转换为char*类型,如果len不为空,同时还设置len为字符串长度;该函数返回的指针,指向的是Lua虚拟机内部的字符

    串,这个字符串是以''结尾的,但字符串中间也可能包含值为0的字符。

    lua_tostring 等价于参数len=NULL时的lua_tolstring;

    lua_tonumber 将给定index索引处的元素转换为double类型;

    lua_topointer 将给定index索引处的元素转换为void*类型;

    lua_tothread 将给定index索引处的元素转换为lua_State*类型(即一个thread);

    lua_touserdata 返回给定index索引处的userdata对应的内存地址;

    thread 操作

    lua_State *lua_newthread (lua_State *L);

    int lua_resume (lua_State *L, int narg);

    int lua_yield (lua_State *L, int nresults);

    lua_newthread 创建一个新的thread,然后压入堆栈,并返回一个lua_State*指针表示创建的新thread。

    新创建的thread与当前thread共享一个全局环境。没有销毁thread的显式调用,它由垃圾收集器负责回收。

    C 调用 Lua代码

    一个简单的例子:

    // test.c

    #include        <stdio.h>

    #include        "lua.h"

    #include        "lualib.h"

    #include        "lauxlib.h"

    /*the lua interpreter*/

    lua_State* L;

    int luaadd(int x, int y)

    {

        int sum;

        lua_getglobal(L,"add");

        lua_pushnumber(L, x);

        lua_pushnumber(L, y);

        lua_call(L, 2, 1);

        sum = (int)lua_tonumber(L, -1);

        lua_pop(L,1);

        return sum;

    }

    int main(int argc, char *argv[])

    {

        int sum;

        L = lua_open();

        luaL_openlibs(L);

        luaL_dofile(L, "add.lua");

        sum = luaadd(10, 15);

        printf("The sum is %d ",sum);

        lua_close(L);

        return 0;

    }

    注意:在C代码里面我们要引入三个头文件lua.h,lauxlib.h和lualib.h:

    lua.h中定义的是最基础的API;

    lauxlib.h中的函数都以luaL_开头,他们是比基础API更抽象的函数;

    lualib.h中定义了打开标准类库的API,比如luaL_openlibs(L)。

    程序开始用luaL_open()函数创建一个lua_State。lua_State中保存了Lua运行时的所有的状态信息(比如变量的值等),并且所有的Lua的C的API都有一个lua_newstate指针的参数。luaL_open函数会创建一个全新的Lua运行时状态,其中没有任何预先定义好的函数(包括最基本的print函数)。如果需要试用标准类库的话,只要调用luaL_openlibs(L)函数就打开标准类库就可以了。标准类库被分别封装在不同的包中,当你需要使用的时候再引入到代码中,这样做的好处是可

    以使Lua尽可能的小(嵌入式语言必须要小),从而可以方便嵌入到其他语言中去。当Lua运行时状态和标准类库都准备完成后,就可以调用luaL_dofile

    (L,"test.lua")函数来执行Lua脚本。运行结束后,需要调用lua_close(L)来关闭Lua运行时状态。

    被调用的test.lua文件:

    -- test.lua

    function add(x,y)

       return x + y

    end

    编译命令,实际命令需要根据自己的lua环境调整

    gcc test.c -o test -llua-5.1 -I /usr/local/include/

    执行./test命令的输出:

    The sum is 25

    另外一个操作table的例子:

    int main()

    {

        lua_State *L = luaL_newstate();

        luaL_openlibs(L); 

        lua_newtable(L); 

        lua_pushstring(L, "i am key");

        lua_pushstring(L, "i am value");

        lua_settable(L, -3); 

        lua_pushstring(L, "i am key");

        lua_gettable(L, -2);

        const char *str = lua_tostring(L, -1);

        printf("%s", str);

        lua_close(L);

        return 0;

    }

    Lua 调用 C 函数

    当Lua调用C函数时,也使用了一个与C语言调用Lua时相同的栈。C函数从栈中获取函数参数,并将结果压入栈中。为了在栈中将函数结果与其他值区分开,C函数还

    应返回其压入栈中的结果个数。

    栈不是一个全局性的结构,每个函数都有自己的局部私有栈。当Lua调用一个C函数时,第一个参数总是这个局部栈的索引1。

    对于可被Lua调用的C函数而言,其接口必须遵循Lua要求的形式,即

    typedef int (*lua_CFunction)(lua_State* L);

    接收一个参数Lua_State*,即Lua的状态,返回值表示压入栈中的结果个数。

    把要调用的C 函数注册到lua状态机中:

    void lua_register (lua_State *L, const char *name, lua_CFunction f);

    lua_register 是一个宏:#define lua_register(L,n,f) (lua_pushcfunction(L, f), lua_setglobal(L, n))

    其中,参数name是lua中的函数名,f 是C中的函数。

    从宏定义可以看出,这个函数的作用是把C函数压入堆栈,并在全局环境中设置Lua函数名;

    Lua在require模块的时候,除了搜索 "*.lua" 文件,也会搜索 "*.so" 文件,也就是说,Lua支持加载C/C++语言编译的动态库文件。

    // foo.c

    #include        "lua.h"

    #include        "lualib.h"

    #include        "lauxlib.h"

    static int add(lua_State *L)

    {

        int n = lua_gettop(L);      /* number of arguments */

        lua_Number sum = 0;

        int i;

        for (i = 1; i <= n; i++) {

            if (!lua_isnumber(L, i)) {

                lua_pushstring(L, "incorrect argument");

                lua_error(L);

            }  

            sum += lua_tonumber(L, i);

        }  

        lua_pushnumber(L, sum/n);    /* first result */

        lua_pushnumber(L, sum);      /* second result */

        return 2;                    /* number of results */

    }

    int luaopen_foo(lua_State *L)

    {

        lua_register(L, "add", add);

        return 1;

    }

    注意:luaopen_MODULE 函数的后缀是有规则的,必须是模块名称,而lua_register的第二个参数是供Lua代码调用的函数名称,第三个参数是当前C函数;

    OK,现在把C代码编译成动态库:

    gcc foo.c -shared -fPIC -o foo.so  -llua-5.1 -I /usr/local/include/

    然后在lua代码里面可以加载该模块:

    require("foo")

    这条命令会自动加载foo.so库,并调用其中的 luaopen_foo 函数,然后执行里面的函数注册代码,这样接下来就能直接使用那些注册到Lua状态机里面的C函数了。

    print(add(14,25,15))

    输出结果:

    18      54

  • 相关阅读:
    [sql]在case语句中不同情况下then的数据的数据类型不一致ORA-00932: inconsistent datatypes: expected NUMBER got CHAR
    环境迁移 小记
    linux下安装oracle遇到的问题
    正向代理与反向代理
    文件夹与SVN脱离关系
    shell 脚本中$$,$#,$?
    在MySQL中单列索引和联合索引的区别
    Java中Map、HashMap、LinkedHashMap、TreeMap的区别
    Error、Exception与RuntimeException的区别
    设计模式--单例模式
  • 原文地址:https://www.cnblogs.com/gd-luojialin/p/10962834.html
Copyright © 2011-2022 走看看