本地声明的lua函数每次传递时都会被解析吗?

假设我在另一个函数中有一个局部声明的函数,该函数经常被调用:

function outer()
    -- 做一些事情
    local loop = true
    while loop do -- 出于某些原因
        local function inner()
            -- 做一些函数的事情
        end
        -- 做一些事情
        inner()
    end
end

每次在运行时遇到内部函数时,它的创建,解析,处理,所有内容都会重新创建吗?基本上,这种模式和声明一个单独的独立函数之间是否存在任何性能差异?

function inner()
    -- 做一些事情
end

functions outer()
    local loop = true
    while loop do
        inner()
    end
end
点赞
用户3574628
用户3574628

简短回答:不行。

当你运行 Lua 代码块(代码块通常指文件,但有时也指字符串)时,在执行之前,整个代码块都会被编译成字节码。当你定义一个局部函数时,你会将对该函数字节码的引用赋值给该函数的名称。Lua 不会重新评估函数本身。

在你的第一段代码示例中,将字节码引用赋给局部变量可能会带来微小的性能损失。当然,你需要进行基准测试以确认。我不知道 Lua 是否可以优化掉代码中这一部分。

请注意闭包可以有不同的上值集,但仍然引用相同的函数字节码。

2021-08-07 06:33:53
用户2047576
用户2047576

除了luther提供的答案外,这里还有一点需要注意,但是你可能会想它是否重要。我认为它不会对性能产生很大影响。

Lua会编译每个函数块,但当你使用一个本地函数时,它将不得不执行CLOSURE指令和MOVE指令来调用特定代码。MOVE被添加是为了防止CALL指令从堆栈中弹出闭包。我不认为Lua在任何情况下都会优化这一点。

如果你使用全局函数或upvalue,它只需要使用GETGLOBAL、GETTABUP或GETUPVALUE指令获取该值。

你可以在这里看到生成的字节码; https://luac.nl/s/473abd2a0bb427a31766ea917

2021-08-09 09:31:26