Answers:
冒号用于实现self
作为第一个参数传递的方法。因此x:bar(3,4)
应与相同x.bar(x,3,4)
。
self
对象将作为第一个参数及其属性值。
object.method(object,args)
检索object
,而一次仅object:method(arg)
检索object
一次。如果object
是全局字段,高:
值字段或表字段,则比快.
。.
永远不会比:
。
对于定义,它与手动指定self 完全相同 -甚至在编译时会产生相同的字节码。即function object:method(arg1, arg2)
与function object.method(object, arg1, arg2)
。
关于使用:
是几乎一样的.
-一种特殊的调用将在内部使用,以确保object
和计算/访问的任何可能的副作用是只有一次计算。呼叫object:method(arg1, arg2)
与相同object.method(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.method(obj, etc)
和之间有明显的区别obj:method(etc)
。