Lua中.和:的区别

246

我对通过.和通过:调用函数的区别感到困惑。

local 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

: 这个符号是干什么用的?


1
相关:https://dev59.com/YVDTa4cB1Zd3GeqPLK11 - finnw
3个回答

345

冒号用于实现将self作为第一个参数传递的方法。所以x:bar(3,4)应该等同于x.bar(x,3,4)


94
啊…所以它是面向对象的语法糖。 - Jason S
15
没错,在整个参考手册中,他们只简要介绍了这个内容:“冒号语法用于定义方法,即具有隐式额外参数self的函数。” (5.0 手册,pdf 第19页底部) - BMitch
4
哦啊...我本来想问这个官方文件在哪里,但是你比我先找到了。做得好 :-) - Jason S
1
@keyle 这取决于 self 对象将作为第一个参数和其属性值。 - user5066707
12
如果调用的对象不是局部变量,使用冒号语法会更快一些,因为虚拟机只需要检索一次。基本上,类似于 object.method(object,args) 的点语法会检索 object 两次,而 object:method(arg) 只会检索一次。如果 object 是全局变量、上值或表字段,则 :. 更快。. 永远不会比 : 快。 - negamartin
显示剩余5条评论

48

对于定义来说,使用冒号与手动指定self完全相同-即在编译时会产生相同的字节码。即function object:method(arg1, arg2)function object.method(self, arg1, arg2)是相同的。

在使用上,冒号几乎相同——内部将使用一种特殊的调用方式,以确保计算/访问object及其任何可能的副作用仅被计算一次。调用object:method(arg1, arg2)否则与object.method(object, arg1, arg2)相同。


35

准确来说,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)之间存在明显的差异。


2
你真的不应该担心这样的事情。如果你必须要担心,那么你的架构肯定出了严重的问题。 - val is still with Monica
8
我认为恰恰相反;良好的代码不应该对无关代码的实现细节做任何假设。函数调用可能被记忆化,也可能没有,这并不意味着经常调用它们是一种好习惯。 - DarkWiiPlayer

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接