Lua C API自定义打印函数,在字符串中传递空格时不被调用

问题描述: 我创建了一个自定义的C++函数 print() ,它应该被推送为一个全局函数,以便用户可以使用 print() 函数来打印到调试控制台。这个函数在某种程度上起作用,但是,当你尝试打印一个带有空格的字符串(超过一个单词)时,这个函数根本不会被调用……这让我很困惑,因为我不知道原因。如果我尝试调用诸如 print("Hello!") 这样的东西,控制台上将打印出 "Hello!",但如果我尝试打印诸如 print("Hello world!") 这样的东西,函数将不会被调用,我知道这是因为我已经使用消息框警告函数何时被调用。

附加信息: 所以,我能找到的最接近的问题就是一个询问如何在Lua C API中使用C++创建自定义打印函数,然后将其推入全局表中的问题。我已经能够做到这一点,我的函数在某种程度上可以工作。我的函数并没有被推到Lua C API的全局表中,而是被推到了一个由我使用lua_newtable(L,s);创建的表中。无论我使用的方式是哪种方式,都没有任何区别。这个打印函数目前不支持表或函数,我只关注找出并修复为什么这个函数无法打印超过一个单词的字符串。以防你想知道,这是使用Lua v5.1.5和Microsoft Visual Studio 2017进行的。调试模式,x86。

Code (C++):

如果有人能帮我修复这个问题,那就太好了!

#include <iostream>
#include <string>
#include <Windows.h>
#pragma comment(lib, "Lua.lib")
#include "lua.hpp"
#include "luaconf.h"

static int print(lua_State* LUASTATE)
{
    MessageBoxA(NULL, "Custom print called.", "FUNCTION!", NULL);
    int nargs = lua_gettop(LUASTATE);
    std::string string = "";
    for (int i = 1; i <= nargs; i++)
    {
        if (i > 1) string += " ";
        switch (lua_type(LUASTATE, i))
        {
        case LUA_TSTRING:
            string += (std::string)lua_tostring(LUASTATE, i);
        case LUA_TNUMBER:
            string += (int)lua_tonumber(LUASTATE, i);
        case LUA_TBOOLEAN:
            string += (bool)lua_toboolean(LUASTATE, i);
        }
    }
    std::cout << string << "\n";
    return 0;
}
int pushs(lua_State* LuaState)
{
    luaL_openlibs(LuaState);
    lua_newtable(LuaState);
    lua_pushcfunction(LuaState, print);
    lua_setglobal(LuaState, "print");
    lua_settop(LuaState, 0);
    return 0;
}
int main()
{
    lua_State* ls = luaL_newstate();
    lua_State* LS = lua_newthread(ls);
    pushs(LS);
    while (true)
    {
        std::cout << " ";
        std::string inputo;
        std::cin >> inputo;
        luaL_dostring(LS, inputo.c_str());
        lua_settop(LS, 0);
    }
    lua_close(LS);
    return 0;
}
点赞
用户9822560
用户9822560

主要问题

std::cin >> inputo 无法从标准输入中读取完整一行,它只能读取一个单词。 因此,当您在 shell 中键入以下输入行时:

print("Hello world")

您的主循环将其分成两个单独的字符串:

  • print("Hello
  • world")

而且这些字符串独立地被 Lua 解释器评估。 这些字符串都不是有效的 Lua 语句,因此解释器不会执行它们。 lua_dostring 将返回错误代码,并在 Lua 栈上留下错误消息。

为了按行在标准输入上工作,您可以使用 std::getline,它在循环中工作得很好:

std::string line;
while (std::getline(std::cin, line)) {
    // do something with line.
}

副注

下面的内容与您的错误没有直接关系,但可能存在问题:

  • std::string += int(或 bool)会将 int 解释为单个 char,并将该单个字符附加到字符串中。
  • 您的 switch/case 似乎缺少 break 语句。
  • lua_State* ls 从未被关闭。
2018-06-12 09:46:40