這篇文章主要介紹了Lua面向對象之類和繼承淺析,本文講解了一些Lua面向對象的一些知識,需要的朋友可以參考下
Lua中的table就是一種對象,但是如果直接使用仍然會存在大量的問題,如下:
代碼如下:
Account = {balance = 0}
function Account.withdraw(v)
Account.balance = Account.balance - v
end
--下面是測試調用函數
Account.withdraw(100.00)
在上面的withdraw函數內部依賴全局變量Account,一旦發生改變,將會導致withdraw不能正常工作,如:
代碼如下:
a = Account; Account = nil
a.withdraw(100.00) --將會導致訪問空nil的錯誤。
這種行為明顯違反了面向對象封裝性和實例獨立性。要解決這一問題,我們需要給withdraw函數再添加一個參數self,他等價於java/C++中的this,如下:
代碼如下:
function Account.withdraw(self,v)
self.balance = self.balance - v
end
--下面是基於修改後代碼的調用:
a1 = Account; Account = nil
a1.withdraw(a1,100.00) --正常工作。
針對上述問題,lua提供了一種更為便利的語法,即將點(.)替換為冒號(:),這樣可以在定義和調用函數時隱藏參數。如:
代碼如下:
function Account:withdraw(v)
self.balance = self.balance - v
end
--調用代碼可改為:
a:withdraw(100.00)
1、類:
lua中再語言上並沒有提供面向對象的支持,因此想實現該功能,我們只能通過table來模擬,如下:
代碼如下:
--這裡的lovenumber是一個公有成員變量
Father={ lovenumber=0}
--new可以視為構造函數
function Father:new(p)
p=p or {} --如果參數中沒有提供table,則創建一個空table
--將新對象實例的元表指向Father,這樣就可以以Father為模板了
setmetatable(p,self)
--將Father的__index字段指向自己,以便新對象在找不到指定的key時可以被重定向,即訪問Father擁有的key
self.__index=self
return p
end
function Father:toString()
print("I love my son!")
end
--Loving被視為公有成員函數
function Father:Loving(v)
self.lovenumber=self.lovenumber+v --這裡的self表示實例對象本身
return self.lovenumber
end
f1=Father:new{name="jianjian"}
f2=Father:new{name="baba",}
print(f1:Loving(100))
print(f2:Loving(200))
--輸出答案
--100
--200
2、繼承
繼承也是面向對象中一個非常重要的概念,在lua中我們也可以像模擬類那樣來實現繼承機制。
代碼如下:
Father={ lovenumber=0}
function Father:new(p)
p=p or {}
--將新對象實例的元表指向Father,這樣就可以以Father為模板了
setmetatable(p,self)
--將Father的__index字段指向自己,以便新對象在找不到指定的key時可以被重定向,即訪問Father擁有的key
self.__index=self
return p
end
function Father:toString()
print("I love my son!")
end
function Father:Loving(v)
self.lovenumber=self.lovenumber+v
return self.lovenumber
end
--下面派生出Father的一個子類,此時的Son仍為Father的一個對象實例
Son=Father:new()
--重寫Father中的toString方法,以實現自定義功能
function Son:toString()
print("I love myself!")
end
--在執行下面的new方法時,table s的元表已經是Son了,而不是Father
s=Son:new()
print(s:toString()) --先在子類Son中找到該方法
print(s:Loving(50)) --子類中無該方法,則調用父類中該方法
--輸出答案
--I love myself!
--50