Lua - 为什么函数调用后的字符串是允许的?
我试图实现一个简单的 C++ 函数,检查 Lua 脚本的语法。为此,我使用了 Lua 的编译器函数 luaL_loadbufferx(),然后检查它的返回值。
最近,我遇到了一个问题,因为我认为应该标记为无效的代码没有被检测到,相反,脚本在运行时失败了(例如在 lua_pcall() 中)。
示例 Lua 代码(可在 official Lua demo 上测试):
function myfunc()
return "everyone"
end
-- 非预期行为的示例:
-- 以下行不会有错误通过编译时检查
print("Hello " .. myfunc() "!") -- 运行时错误:企图调用一个字符串值
print("Hello " .. myfunc() {1,2,3}) -- 运行时错误:企图调用一个字符串值
-- 其他示例:
-- 以下行包含无效语法的示例,编译器会检测到。
print("Hello " myfunc() .. "!") -- 编译错误:')'预期在 'myfunc' 附近
print("Hello " .. myfunc() 5) -- 编译错误:')'预期在‘5’附近
print("Hello " .. myfunc() .. ) -- 编译错误:')' 附近存在意外符号
显然,目标是在编译时捕获所有语法错误。所以我的问题是:
- 什么是“调用字符串值”指的是什么?
- 为什么首先允许这种语法?这是我不知道的某种 Lua 特性,还是
luaL_loadbufferx()在这个特定的示例中有问题? - 是否可能通过其他方法检测到此类错误而不运行它?不幸的是,我的函数在编译时没有访问全局变量的权限,因此我不能直接通过
lua_pcall()运行代码。
注意:我使用的是 Lua 版本 5.3.4(手册)。
非常感谢您的帮助。
我写下这篇回答来帮助未来遇到类似问题的人,同时也为他们提供解决方案。
手册
Lua 手册(在其 3.4.10 节中 —— 函数调用) 基本上表明存在三种不同的方法来向 Lua 函数提供参数。
参数具有以下语法:
args ::= ‘(’ [explist] ‘)’ args ::= tableconstructor args ::= LiteralString所有参数表达式都在调用之前被计算。f{fields} 形式的调用是 f({fields}) 的语法糖;即参数列表是一个单独的新表。f'string'(或 f"string" 或 f[[string]]形式的调用是 f('string') 的语法糖;即参数列表是一个单独的字面量字符串。
解释
正如 lhf 在 他的答案 中指出的,myfunc()"!" 和 myfunc(){1,2,3} 都是有效的 Lua 表达式。这意味着 Lua 编译器没有做错任何事情,因为它在编译时不知道函数返回值。
问题中给出的原始示例代码:
print("Hello " .. myfunc() "!")
然后可以改写为:
print("Hello " .. (myfunc()) ("!"))
这将在执行时转换为:
print("Hello " .. ("everyone") ("!"))
从而导致运行时错误消息 attempt to call a string value(可以改写为:字符串 everyone 不是一个函数,所以你不能调用它)。
解决方案
据我理解,这两种替代的参数提供方式与标准的 func(arg) 语法没有真正的优势。这就是为什么我最终修改了 Lua 解析器文件。保留此替代语法的缺点太大了。以下是我所做的(适用于 v5.3.4):
在
lparser.c文件中搜索以下函数:static void suffixedexp (LexState *ls, expdesc *v)在此函数中更改以下 case 语句:
case '(': case TK_STRING: case '{':更改为:
case '(':
警告!通过这样做,我修改了 Lua 语言,因此正如 lhf 在他的评论中所述,它不再被称为_纯_ Lua。如果您不确定是否正是您想要的,我不推荐这种方法。
通过这个 _小小的修改_,编译器将检测到上述的这两种替代语法,并将其视为错误。当然,我不能再在 Lua 脚本中使用它们,但对于我的特定应用程序来说就足够了。
我需要做的一切就是将这个变化 _记录在某个地方_,以便在升级到更高的 Lua 版本时找到它。
另一个方法是改变字符串的元表,使调用字符串成为有效的。
local mt = getmetatable ""
mt.__call = function (self, args) return self .. args end
print(("x") "y") -- 输出 `xy`
现在对字符串的有效语法调用将导致字符串连接,而不是运行时错误。
- Lua 虚拟机加密load(string.dump(function)) 后执行失败问题如何解决
- 我想创建一个 Nginx 规则,禁止访问
- 如何将两个不同的lua文件合成一个 东西有点长 大佬请耐心看完 我是小白研究几天了都没搞定
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?

myfunc() "!"和myfunc(){1,2,3}都是有效的 Lua 表达式。Lua 允许形如 exp string 的函数调用。详见Lua 语法中的
functioncall和prefixexp。因此,
myfunc() "!"是一个有效的函数调用,调用myfunc返回的任何内容,并将其与字符串"!"一起调用。对于形如 exp table-literal 的函数调用,同样的事情也会发生。