使用闭包方法实现 OOP 时,如何实现受保护的成员?

我目前正在使用闭包在Lua中实现面向对象编程。以下是一个简化的示例。我的问题出现在尝试实现infested_mariner内部的stronger_heal时发生。

如果我将我的infested_mariner定义放在另一个.lua文件中,则它将无法访问全局私有变量,也无法访问基本.lua文件中定义的私有变量。如何拥有受保护的成员,只有“infested_mariner”可以访问,并且解决方案不涉及所有派生类都在与父类相同的文件中?

注意:我目前在子类中使用getter和setter。

点赞
用户90511
用户90511

在Lua中,您只能在其作用域内访问局部变量。为了使其他函数可以看到您的变量,需要对其进行重写,使受保护的变量位于某个可被子类访问的表中。

一种方法是只需在当前类中创建公共属性,并使用命名约定(例如以下划线开头的名称)来表示受保护的内容。您可能已经知道这一点,但我必须说,我认为这种方法通常比真正的受保护变量更简单。

如果您想要真正的受保护变量,需要将公共内容和受保护内容分开放在两个表中。其中一种方法是更改bless函数,以使其接收这两个表:

function infested_mariner.bless (pub, pro)
   -- New methods:
   function pub.strongerheal (value)
     pro.hp = pro.hp + value*2
   end
   return pub
end

如何设置构造函数以便它们将受保护的表传递给彼此的方法留作练习。如果您选择这个方法,您可能希望有一些函数能够为您完成它,以便您不必每天都有机会操纵受保护的表。

2013-05-26 15:44:49
用户856565
用户856565

这是闭包方法的局限性。无法从闭包外部访问 hp

这有点是一种思想问题……一些人认为,保护成员会破坏封装性,因此应该避免使用:将所有数据设置为私有,并在需要时扩展接口(添加公共函数)。这样说的人往往喜欢闭包方法。

如果您想使用保护成员,我可能不会使用闭包方法。良好的命名约定可能是最简单的方法。

self.public_thing = "asdf"
self._protected_thing = "asdf"

当然,数据实际上是公共的,但这种方法相当有效。这基本上是 Python 所做的。当您想搞砸内部时,例如进行测试时,这很有用。

2013-07-19 18:53:32