首页 > 解决方案 > Lua C API 自定义打印函数,字符串中传入空格时不调用

问题描述

问题描述: 我创建了一个自定义 C++ 函数print(),它应该作为全局推送到表中,以便用户可以使用该print()函数打印到调试控制台。这个函数在某种程度上有效,但是,当你尝试打印一个带有空格的字符串(超过一个单词)时,这个函数根本不会被调用......这让我非常困惑,因为我不知道为什么。如果我尝试调用诸如print("Hello!")控制台之类的东西,将会有“你好!” 打印到它,但是如果我尝试打印诸如print("Hello world!")根本不会调用该函数之类的东西,我知道这一点,因为我使用了一个消息框来在调用该函数时发出警报。

附加信息: 因此,我能找到的最接近的问题是询问如何使用 Lua C API 在 C++ 中制作自定义打印函数,然后将其推送到全局表上。我已经可以做到这一点,并且我的功能在一定程度上起作用。我的函数没有被推送到 Lua C API 的全局表中,而是被推送到我使用lua_newtable(L, s);. 但是,两种方法我都试过了,没什么区别。到目前为止,此打印功能不支持表格或功能,我只是专注于找出并修复该功能无法在一个单词上打印字符串的原因。以防万一您想知道,为此使用了 Lua v5.1.5 和 Microsoft Visual Studio 2017。调试模式,x86。

代码(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;
}

标签: c++luac

解决方案


主要问题

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 解释为 single char,并将这个单个字符附加到字符串中。
  • switch/case似乎缺少break陈述。
  • lua_State* ls永远不会关闭。

推荐阅读