我对via函数调用之间的区别感到困惑。通过:
> x = {foo = function(a,b) return a end, bar = function(a,b) return b end, }
> return x.foo(3,4)
3
> return x.bar(3,4)
4
> return x:foo(3,4)
table: 0x10a120
> return x:bar(3,4)
3
在做什么?
我对via函数调用之间的区别感到困惑。通过:
> x = {foo = function(a,b) return a end, bar = function(a,b) return b end, }
> return x.foo(3,4)
3
> return x.bar(3,4)
4
> return x:foo(3,4)
table: 0x10a120
> return x:bar(3,4)
3
在做什么?
对于定义来说,它与手动指定self完全相同——它甚至会在编译时产生相同的字节码。即函数对象:method(arg1, arg2)与函数对象相同。方法(self, arg1, arg2)。
在使用上:几乎是一样的。一种特殊的调用将在内部使用,以确保对象和任何可能的副作用的计算/访问只计算一次。调用object:method(arg1, arg2)在其他方面与object相同。方法(对象,arg1, arg2)。
为了完全精确,obj:method(1,2,3)与
do
local _obj = obj
_obj.method(_obj, 1, 2, 3)
end
为什么是局部变量?因为,正如许多人指出的那样,obj:method()只索引_ENV一次来获得obj。在考虑速度时,这通常很重要,但考虑以下情况:
local tab do
local obj_local = { method = function(self, n) print n end }
tab = setmetatable({}, {__index = function(idx)
print "Accessing "..idx
if idx=="obj" then return obj_local end
end})
end
tab.obj.method(tab.obj, 20)
--> Accessing obj
--> Accessing obj
--> 20
tab.obj:method(10)
--> Accessing obj
--> 10
现在想象一下__index元方法不仅仅是打印一些东西。想象一下,它增加了一个计数器,将一些内容记录到文件中,或者从数据库中删除了一个随机用户。做两次和只做一次是有很大区别的。在这种情况下,obj之间有明显的区别。方法(obj, etc)和obj:方法(etc)。