调用 Lua 函数参数时使用 unpack()会发生什么?

以下是我看到的:

> function test(...)
>>  for i, v in ipairs({...}) do
>>    print(v)
>>  end
>>end

-- 这是预期的
> test(unpack({1,2}))
1
2
-- 当我混合使用解包和常规参数时
-- 2 发生了什么?
> test(unpack({1,2}), 3)
1
3
-- 3 发生了什么?
> test(unpack({}), 3)

-- 当我在解包之前放置常规参数时,一切都很好:
> test(1, unpack({2, 3}))
1
2
3

当我混合使用解包参数和常规参数时,结果很奇怪。你能解释一下底层发生了什么吗?

点赞
用户5675002
用户5675002

在 Lua 手册中有如下解释:

如果表达式作为表达式列表的最后一个(或是唯一的)元素,那么就不做任何调整(除非将表达式放在括号中)。 在其他所有上下文中,Lua 将结果列表调整为一个元素,只保留第一个元素,丢弃所有其它的值

你得到的就是这个结果。unpack() 返回多个值,这些返回的元素列表将被调整为 1 个元素,除非它是 test() 参数中的最后一个表达式。

2016-05-22 09:08:52
用户2858170
用户2858170

我引用Lua参考手册:

函数调用和可变参数表达式都可以产生多个值。如果一个表达式被用作语句(只有对函数调用(见§2.4.6)才可能),那么它的返回列表将调整为零元素,从而丢弃所有返回值。如果一个表达式被用作列表的最后(或仅)一个元素,则不进行调整(除非调用被括在括号中)。在所有其他语境中,Lua将结果列表调整为一个元素,丢弃除第一个元素外的所有值。

正如您所看到的,您的拆分调用被减少为一个返回值,因为它既不是在您传递给test的表达式列表中的最后一个,也不是唯一的表达式:

test(unpack({1,2}), 3)

在另一种情况下,答案非常简单:

test(unpack({}), 3)

传递给test的第一个值为nil。因此,由于您的表的第一个值是nil,for i,v in ipairs({...}) do end将什么也不做,因为unpack({})返回nil

ipairs(t)

返回三个值(一个迭代器函数,表t和0),以便构造

for i,v in ipairs(t) do body end

将迭代键-值对(1,t[1]),(2,t[2]),...,直到第一个nil值。

2016-05-22 09:09:08