【问题标题】:From C, how can I print the contents of the Lua stack?从 C 中,我如何打印 Lua 堆栈的内容?
【发布时间】:2020-03-24 06:21:04
【问题描述】:

我的 C 程序可能有一个愚蠢的错误。 Lua 堆栈在某些情况下不包含我认为它应该包含的值。

为了调试它,我想在程序的某个点打印 Lua 堆栈的内容。我怎样才能做到这一点而不会弄乱过程中的堆栈?

【问题讨论】:

标签: c lua lua-api


【解决方案1】:

此答案是@lhf 在 cmets 中提供的答案的略微编辑版本。

它的优点是不修改堆栈中的任何值,并且不需要任何额外的空间。

static void dumpstack (lua_State *L) {
  int top=lua_gettop(L);
  for (int i=1; i <= top; i++) {
    printf("%d\t%s\t", i, luaL_typename(L,i));
    switch (lua_type(L, i)) {
      case LUA_TNUMBER:
        printf("%g\n",lua_tonumber(L,i));
        break;
      case LUA_TSTRING:
        printf("%s\n",lua_tostring(L,i));
        break;
      case LUA_TBOOLEAN:
        printf("%s\n", (lua_toboolean(L, i) ? "true" : "false"));
        break;
      case LUA_TNIL:
        printf("%s\n", "nil");
        break;
      default:
        printf("%p\n",lua_topointer(L,i));
        break;
    }
  }
}

如果需要,您还可以在 LUA_TNUMBER 情况下使用 lua_isinteger(L, i) 以区分整数和浮点数。

【讨论】:

    【解决方案2】:

    这段代码从上到下遍历栈,对每个值调用tostring,打印结果(如果没有得到结果,打印类型名)。

    assert(lua_checkstack(L, 3));
    int top = lua_gettop(L);
    int bottom = 1;
    lua_getglobal(L, "tostring");
    for(int i = top; i >= bottom; i--)
    {
        lua_pushvalue(L, -1);
        lua_pushvalue(L, i);
        lua_pcall(L, 1, 1, 0);
        const char *str = lua_tostring(L, -1);
        if (str) {
            printf("%s\n", str);
        }else{
            printf("%s\n", luaL_typename(L, i));
        }
        lua_pop(L, 1);
    }
    lua_pop(L, 1);
    

    【讨论】:

    • 您忘记检查 Lua C API 堆栈中的可用空间
    • 不幸的是,这个解决方案没有保留L-&gt;top 的值,所以它对我不起作用。
    • @hugomg 不是吗?当代码结束时,它应该有初始值。
    • 此代码的固定版本必须确保有足够的空间来推送额外的元素,这可能需要重新分配堆栈。没错,大多数时候这无关紧要,但这确实意味着,如果您尝试调试涉及 Lua 堆栈调用 thid 打印函数的崩溃,则可能会改变您尝试调试的程序的行为。
    • 这段代码改变了堆栈中的值:它将数字转换为字符串。
    猜你喜欢
    • 2012-02-25
    • 2020-08-28
    • 1970-01-01
    • 1970-01-01
    • 2012-08-28
    • 2015-11-05
    • 2017-08-15
    • 2010-09-08
    • 2023-04-06
    相关资源
    最近更新 更多