当前位置: 首页 > 工具软件 > Lua Fun > 使用案例 >

lua c api

国晟睿
2023-12-01
lua_Stat* lua = luaL_newstate(); // 一个新的 lua 状态机
luaL_openlibs( lua ); // 加载标准lib

int my_fun_name( lua_State *L ) {
    int argc = lua_gettop( L ); // 获得函数参数个数
    for( int i=1; i <= argc; ++i ) { // 注意 = 
        // 以下函数使用的参数序号规则如下:
        //   调用参数: A  B  C  D
        //   正序序号: 1  2  3  4
        //   反序序号:-4 -3 -2 -1
        // 但是参数入栈却是:
        //   --------->
        //   D  C  B  A
        // lua_toboolean
        // lua_tocfunction
        // lua_tointeger 这是 ptrdiff_t,表示为机器最大整数类型
        // lua_tonumber  这是 double
        // lua_topointer
        // lua_tostring 等价于 lua_tolstring ,而参数 len 设为 NULL
        // lua_tolstring
        // lua_tothread
        // lua_touserdata userdata 和 light userdata 公用这个函数
        // 将指定位置的参数转成C语言数值格式(需要它本身就是数值,否则返回0 )
        printf( "lua fun arg[index: %d, type: %d, type_name: %s] : %lf\n", 
            i, lua_type( L, i ), lua_typename( L, i ), lua_tonumber( L, i ) ); 
    }
    // lua_to* 函数只获得参数值,参数此时还在栈上
    lua_pop( L, argc ); // 弹出所有参数
    
    // 创建一个新的 table
    lua_newtable( L );  // 栈 -3
    lua_pushstring( L, "name" );   // 栈 -2 key
    lua_pushstring( L, "参数啊" ); // 栈 -1 value
    // 另一种形式:lua_setfield( L, -3, "name" ),通过参数这是 key
    lua_settable( L, -3 ); // 通过栈来表示k&v,这会弹出 key(-2) & value(-1)
    lua_setglobal( L, "t" ); // 弹出 栈-3(table),并将值设置到全局变量 't' 中
    // 此时栈平衡
    luaL_dostring( L, "print(t.name)" ); // 输出:参数啊
    
    // table 遍历,记住格式就行,只有看源码才能知道为什么这么写
    /* table 放在索引 't' 处 */
    lua_getglobal( L, "t" );
    int t = -2; // -2 是因为要 push 一个 nil
    lua_pushnil(L);  /* 第一个 key */
    while (lua_next(L, t) != 0) {
        /* 用一下 'key' (在索引 -2 处) 和 'value' (在索引 -1 处) */
        printf("%s - %s\n",
        lua_typename(L, lua_type(L, -2)),
        lua_typename(L, lua_type(L, -1)));
        /* 移除 'value' ;保留 'key' 做下一次迭代 */
    lua_pop(L, 1);
    }
    // 要保持栈平衡,移除 table t
    lua_pop(L, 1);
    
    // 调用 lua 函数
    // #define lua_getglobal(L,s)  lua_getfield(L, LUA_GLOBALSINDEX, s)
    // 注意 LUA_GLOBALSINDEX,表示全局范围
    lua_getglobal(L, "f");     // 从全局范围内寻找将调用的函数 f,并入栈
    lua_pushstring(L, "how");  // 第 1 个参数
    lua_getglobal(L, "t");     // 获得 table t 的索引,并压入栈      <-----------------+
    lua_getfield(L, -1, "x");  // 压入 t.x (栈位置-1就是table t) 的值(第 2 个参数     |
    lua_remove(L, -2);         // 从堆栈中移去 't'    ------------------------注意 ---+
    lua_pushinteger(L, 14);    // 第 3 个参数
    lua_call(L, 3, 1);         // 调用 'f',传入 3 个参数,并索取 1 个返回值
    lua_setglobal(L, "a");     // 将最后一个栈数据弹出(也就是函数返回值)并设置到全局变量 'a' 中
    //  table 的索引,为什么称“索引”,记住这个概念即可
    
    // 这些函数都有对应的 check 函数:lua_is*
    lua_pushcclosure( L, NULL, 0 );
    lua_pushcfunction( L, NULL );
    lua_pushlightuserdata( L, NULL ); //  light userdata,就是说你管理这个指针的数据,比如 malloc 要自己 free
    lua_newuserdata( L, 10 /* Byte*/ ); 这是使用 lua 提供的 malloc,lua gc 负责回收
    lua_pushstring( L, "string" );
    lua_pushlstring( L, "定长 string", sizeof("定长 string") );
    lua_pushfstring( L, "[%s] 十分受限的 sprintf", "参数1");
    lua_pushvfstring( L, ...) // vprintf
    lua_pushnil( L );
    lua_pushthread( L ); // ???
    lua_pushvalue( L, 1 ); // 对指定的栈元素拷贝
    lua_pushinteger(L, 99 );  // 第一个返回值
    lua_pushnumber(L, 9.9 );  // 第二个返回值
    
    // 这个函数需要返回两个值,之前有调用 lua_pop,清空了栈,
    // 如果没有清空,lua 也会丢弃不需要的栈数据,比如现在的 push,
    // 但是函数只需要返回2个,除了最后两个(-1,-2)会被保留,其他的都将被丢弃
    return 2;
}

int 闭包函数( lua_State *L ) {
    double upval1 = 0, upval2 = 0;
    /* 注意upvalue索引1,2是闭包依赖的,不会和其他的闭包中的索引冲突 */
    upval1 = lua_tonumber(L, lua_upvalueindex(1) );
    upval2 = lua_tonumber(L, lua_upvalueindex(2) );
    upval1++; upval2++;    
    lua_pushnumber(L, upval1);
    lua_replace(L, lua_upvalueindex(1));/* 更新upvalue1 的值 */
    lua_pushnumber(L, upval2);
    lua_replace(L, lua_upvalueindex(2));/* 更新upvalue2 的值 */
    
    return 0;
}

int 闭包产生器( lua_State *L ) {
    lua_pushstring( lua, "闭包的参数1" );
    lua_pushstring( lua, "闭包的参数2" );
    lua_pushcclosure( lua, 闭包函数, <闭包函数需要的闭包的参数> ); // 弹出所有闭包的参数
    return 1;
}
lua_register( lua, "closure_name", 闭包产生器 );

// 将 fun_name 函数注册为的全局函数
lua_register( lua, "fun_name", my_fun_name );

// 导出函数,可以作为 lua 的扩展库
// 注意"模块名",要一致!!!
int luaopen_模块名(lua_State* lua) {

    static struct luaL_Reg funs[] = {
        {"fun_name", my_fun_name},
        {NULL, NULL},
    };
    luaL_register( lua, "模块名", funs );
    return 1;
}

// 销毁指定 Lua 状态机中的所有对象
lua_close( lua );


//------------------------------------------
//------------------------------------------
//------------------------------------------

int pmain( lua_State* L )
{
	luaL_openlibs( L );
	
	luaL_dostring( L, "print(\"啦啦啦啦\")" );

	return 0;
}

// 按照 风云大佬 的说法是不是该这么写?
// https://blog.codingnow.com/2015/05/lua_c_api.html
int main( int argc, char** argv )
{
	lua_State* lua = luaL_newstate();
	lua_pushcfunction( lua, pmain );
	// 但是为什么stdout 内容终端无法获取,使用 gdb 调试可以获得输出
	printf( "ret: %d\n", lua_pcall( lua, 0, 0, 0 ) );

	return 0;
}

 类似资料:

相关阅读

相关文章

相关问答