我对通过.
和:
> 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
:
在做什么?
答案 0 :(得分:198)
冒号用于实现将self
作为第一个参数传递的方法。因此x:bar(3,4)
应与x.bar(x,3,4)
相同。
答案 1 :(得分:23)
对于定义,与完全与手动指定self相同 - 它甚至会在编译时产生相同的字节码。即function object:method(arg1, arg2)
与function object.method(self, arg1, arg2)
相同。
使用时:
几乎与.
相同 - 我们会在内部使用一种特殊的电话来确保object
和任何可能的方面 - 计算/访问的影响只计算一次。调用object:method(arg1, arg2)
与object.method(object, arg1, arg2)
相同。
答案 2 :(得分:15)
确切地说,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.method(obj, etc)
和obj:method(etc)
之间有明显的区别。