首页 > 代码库 > 【Lua】面向对象编程(一)

【Lua】面向对象编程(一)

类和对象:

account.lua  

module(...,package.seeall)Account={balance=0}Account.new=function(self,o)    local o=o or {}    setmetatable(o,self)    self.__index=self    return oendAccount.getBalance=function(self)    return self.balanceendAccount.setBalance=function(self,money)    self.balance=moneyendAccount.withdraw=function(self,money)    if money<0 then    error("<Account.deposit> money must > 0")    else    if money > self.balance then        error("<Account.deposit> no enough balance")    end    self.balance=self.balance-money    endendAccount.deposit=function(self,money)    if money <0 then    error("<Account.deposit> money must > 0")    end    self.balance=self.balance+moneyendAccount.__tostring=function(self)    str=os.date("%y/%m/%d %H:%M:%S",os.time())    str=string.format("%s\nBalance:%s",str,self.balance)    return strend

 

注:

       Lua检测到没有的字段时,但在其元表中有一个__index字段,那么Lua会调用这个__index元方法,并返回结果,__index元方法可以使函数或者是table,当元方法是一个函数时,Lua以table和不存在的key做为参数来调用该函数,当元方法是一个table时,Lua重新访问这个table。

    上述代码,当调用Account:new 时,self等于Account,因此可以直接使用Account来代替self,当引入类的继承时,使用self更加准确。

 

继承:

specialAccount.lua 

module(...,package.seeall)SpecialAccount=MAccount.Account:new({limit=1000.0})SpecialAccount.getLimit=function(self)    return self.limitendSpecialAccount.withdraw=function(self,money)    if money <0 then    str="<SpecialAccount.withdraw> money must > 0"    Logger:error(str)    error(str)    end    if money > self:getLimit() then    Logger:warn("<SpecialAccount.with> withdraw > limit")    return    end    self.balance=self.balance-moneyendSpecialAccount.__tostring=function(self)    str=string.format("%s\nLimit:%d",MAccount.Account.__tostring(self),self.limit)    return strend

 

说明:

    SpecialAccount继承了Account的所有函数和变量,在执行SpecialAccount的new函数,它的self参数表示为SpecialAccount,因此执行以下语句:

clipboard[3]

     s表示的元表为SpecialAccount,而SpecialAccount中的__index值也是SpecialAccount,s继承自SpecialAccount,而SpecialAccount又继承自Account,Lua在s中找不到deposit字段时,会查找SpecialAccount,如果仍找不到deposit字段,就会查找Account,最终会在那里找到deposit字段。

    SpecialAccount重新定义了基类继承的方法,如SpecialAccount中的withdraw函数,当调用s:withdraw(200)时,Lua就不会再Account中查找了,因为Lua会在SpecialAccount中找到withdraw方法。

 

补充说明:

      Lua中对象有一个特殊现象,无需为指定一种新行为而创建一个新类,如果只有一个对象需要这种特殊的行为,那么可以直接在该对象中实现这个行为,例如,账户s表示一个特殊的客户,这个客户的透支额度总是余额的10%。那么可以这么修改这个对象:

 clipboard[4]

在这段代码后,调用s:withdraw(200)还是会执行SpecialAccount中的withdraw函数,但是withdraw调用的self:getLimit则是上面的定义。

 

完整代码:

模块结构:

image

 

init.lua 

MAccount=require("class.account")MSpecialAccount=require("class.specialAccount")local LoggerConsole=require("logging.console")Logger=LoggerConsole()



main.lua   

local path="C:\\Users\\Administrator\\Desktop 2\\Thinking\\Lua\\"package.path=string.format("%s;%s?.lua;%s?/init.lua",package.path,path,path)require("class")local function main()    local account=MAccount.Account:new()    account:deposit(500)    account:withdraw(125)    print(account)    account=MSpecialAccount.SpecialAccount:new()    print(account:getLimit())    account:deposit(300)    account:withdraw(1200)    print(account)endmain()

 

说明:

        上述修改了package.path 的路径,具体原因参考 本博客“【Lua】模块和包”文章

【Lua】面向对象编程(一)