如何使用重新定义的打印功能打印 Lua table?
How to print Lua table using the redefined print function?
我从这个 post 学会了如何在 C++ 中重新定义 Lua 的 print()
。 ()
这是重新定义的打印函数,可将变量打印到我的主机程序的控制台。 (通过名为 post..
的函数)
int l_my_print(lua_State *L)
{
int nargs = lua_gettop(L);
for (int i = 1; i <= nargs; ++i)
{
if (lua_isnil(L, i))
poststring("nil");
else if (lua_isboolean(L, i))
lua_toboolean(L, i) ? poststring("true") : poststring("false");
else if (lua_isnumber(L, i))
postfloat(static_cast<t_float>(lua_tonumber(L, i)));
else if (lua_isstring(L, i))
poststring(lua_tostring(L, i));
else if (lua_istable(L, i))
poststring("table: "); //how to print like Lua's built-in print()?
}
endpost();
return 0;
}
除了我打印 tables 时,此代码工作正常。 (它现在只打印 table:
)
我想知道如何像 Lua 的 print()
一样打印 table。
比如我在运行下面的代码中Lua:(重定义前)
print({1,2,3});
我得到这个结果:(似乎在不断变化)
table: 0x23b8660
这是指向 Lua-table 的指针的十六进制表示吗?
我应该如何处理我的 l_my_print()
函数才能像 Lua 的 print()
一样工作?
只需使用 luaL_tolstring
即可获取任何内容的字符串表示形式。这也尊重 __tostring
元方法。下面的示例使用 C++17 中的 std::string_view
作为零拷贝只读字符串参数。
#include <iostream>
#include <string_view>
#include <lua.hpp>
void poststring(std::string_view sv) { std::cout << sv << '\n'; }
void endpost() { std::cout << "---\n"; }
int l_my_print(lua_State *L) {
int nargs = lua_gettop(L);
for (int i = 1; i <= nargs; ++i) {
poststring(luaL_tolstring(L, i, nullptr));
lua_pop(L, 1); // remove the string
}
endpost();
return 0;
}
int main() {
lua_State *L = luaL_newstate();
luaL_openlibs(L);
lua_pushcfunction(L, l_my_print);
lua_setglobal(L, "my_print");
int i = 0;
lua_pushlightuserdata(L, &i);
lua_setglobal(L, "udata");
luaL_dostring(L, "my_print(1, 3.14, \"Hello World\")\n"
"my_print(false, udata, {})\n");
lua_close(L);
}
调用示例:
$ clang++ -Wall -Wextra -Wpedantic -std=c++17 -I/usr/include/lua5.3 test.cpp -llua5.3
$ ./a.out
1
3.14
Hello World
---
false
userdata: 0x7fff4685993c
table: 0x883300
---
或者您可以在 Lua 中进行转换:
--Produces a compact, uncluttered representation of a table. Mutual recursion is employed
--source: http://lua-users.org/wiki/TableUtils
function table.tostring( tbl )
local result, done = {}, {}
for k, v in ipairs( tbl ) do
table.insert( result, table.val_to_str( v ) )
done[ k ] = true
end
for k, v in pairs( tbl ) do
if not done[ k ] then
table.insert( result,
table.key_to_str( k ) .. "=" .. table.val_to_str( v ) )
end
end
return "{" .. table.concat( result, "," ) .. "}"
end
我从这个 post 学会了如何在 C++ 中重新定义 Lua 的 print()
。 ()
这是重新定义的打印函数,可将变量打印到我的主机程序的控制台。 (通过名为 post..
的函数)
int l_my_print(lua_State *L)
{
int nargs = lua_gettop(L);
for (int i = 1; i <= nargs; ++i)
{
if (lua_isnil(L, i))
poststring("nil");
else if (lua_isboolean(L, i))
lua_toboolean(L, i) ? poststring("true") : poststring("false");
else if (lua_isnumber(L, i))
postfloat(static_cast<t_float>(lua_tonumber(L, i)));
else if (lua_isstring(L, i))
poststring(lua_tostring(L, i));
else if (lua_istable(L, i))
poststring("table: "); //how to print like Lua's built-in print()?
}
endpost();
return 0;
}
除了我打印 tables 时,此代码工作正常。 (它现在只打印 table:
)
我想知道如何像 Lua 的 print()
一样打印 table。
比如我在运行下面的代码中Lua:(重定义前)
print({1,2,3});
我得到这个结果:(似乎在不断变化)
table: 0x23b8660
这是指向 Lua-table 的指针的十六进制表示吗?
我应该如何处理我的 l_my_print()
函数才能像 Lua 的 print()
一样工作?
只需使用 luaL_tolstring
即可获取任何内容的字符串表示形式。这也尊重 __tostring
元方法。下面的示例使用 C++17 中的 std::string_view
作为零拷贝只读字符串参数。
#include <iostream>
#include <string_view>
#include <lua.hpp>
void poststring(std::string_view sv) { std::cout << sv << '\n'; }
void endpost() { std::cout << "---\n"; }
int l_my_print(lua_State *L) {
int nargs = lua_gettop(L);
for (int i = 1; i <= nargs; ++i) {
poststring(luaL_tolstring(L, i, nullptr));
lua_pop(L, 1); // remove the string
}
endpost();
return 0;
}
int main() {
lua_State *L = luaL_newstate();
luaL_openlibs(L);
lua_pushcfunction(L, l_my_print);
lua_setglobal(L, "my_print");
int i = 0;
lua_pushlightuserdata(L, &i);
lua_setglobal(L, "udata");
luaL_dostring(L, "my_print(1, 3.14, \"Hello World\")\n"
"my_print(false, udata, {})\n");
lua_close(L);
}
调用示例:
$ clang++ -Wall -Wextra -Wpedantic -std=c++17 -I/usr/include/lua5.3 test.cpp -llua5.3
$ ./a.out
1
3.14
Hello World
---
false
userdata: 0x7fff4685993c
table: 0x883300
---
或者您可以在 Lua 中进行转换:
--Produces a compact, uncluttered representation of a table. Mutual recursion is employed
--source: http://lua-users.org/wiki/TableUtils
function table.tostring( tbl )
local result, done = {}, {}
for k, v in ipairs( tbl ) do
table.insert( result, table.val_to_str( v ) )
done[ k ] = true
end
for k, v in pairs( tbl ) do
if not done[ k ] then
table.insert( result,
table.key_to_str( k ) .. "=" .. table.val_to_str( v ) )
end
end
return "{" .. table.concat( result, "," ) .. "}"
end