概述
一個(gè)類就是像是一個(gè)創(chuàng)建對(duì)象的模具,對(duì)于Lua這種沒有類的概念的語言,為了模擬類,方法是為要?jiǎng)?chuàng)建的對(duì)象制定一個(gè)原型(prototype)。這個(gè)原型相當(dāng)于其他語言中的類。但是原型同時(shí)也是一種常規(guī)的對(duì)象,當(dāng)其他的對(duì)象(看成是原型的實(shí)例)遇到一個(gè)未知的操作時(shí),就會(huì)去原型中查找。因此,在Lua這種沒有類的語言中,為了表示一個(gè)類,只需創(chuàng)建一個(gè)專用作其他對(duì)象的原型。類和原型都是一種組織對(duì)象間共享行為的方式。本文將在Lua中模擬類,相關(guān)的代碼可以在我的github上直接運(yùn)行。
實(shí)現(xiàn)
在Lua中要模擬類比較關(guān)鍵的地方就是class的繼承機(jī)制,以及class實(shí)例化的過程,這個(gè)過程的主要是用了元表技術(shù)以及是否把方法拷貝到子類或?qū)嵗校ㄈ艨截悾瑒t增加了數(shù)據(jù)冗余,并且喪失了父類更新子類也會(huì)自動(dòng)更新的特性,若不拷貝,則每次訪問父類方法,由于使用元表,都會(huì)代碼額外的開銷),下面是一種實(shí)現(xiàn)方法:
clsObject = {
__ClassType = "class type"
}
function clsObject:Inherit(o)
o = o or {}
o.__ClassType = "class type"
o.mt = { __index = o}
setmetatable(o, {__index = self})
return o
end
function clsObject:New(...)
local o = {}
setmetatable(o, self.mt)
if o.__init__ then
o:__init__(...)
end
return o
end
function clsObject:__init__()
end
function clsObject:Destroy()
end
function clsObject:GetType()
return "BaseClass"
end
上面,不管在繼承機(jī)制還是實(shí)例化的過程,都是使用了元表技術(shù),這樣做符合class繼承的思想。另外還上面的實(shí)現(xiàn),還可以實(shí)現(xiàn)工具函數(shù):獲取一個(gè)class的父類和判斷一個(gè)class是否是子類或是對(duì)象是否改類的實(shí)例,代碼如下:
function Super(TmpClass)
return getmetatable(TmpClass).__index
end
function IsSub(clsSub, clsAncestor)
local Temp = clsSub
while 1 do
local mt = getmetatable(Temp)
if mt then
Temp = mt.__index
if Temp == clsAncestor then
return true
end
else
return false
end
end
end
可以按下面實(shí)例代碼來使用這個(gè)類
clsParent = clsObject:Inherit()
function clsParent:Foo()
print("ParentFoo!")
end
local ParentObj = clsParent:New()
ParentObj:Foo()
clsSon = clsParent:Inherit()
function clsSon:Foo()
Super(clsSon).Foo(self)
print("SonFoo")
end
local SonObj = clsSon:Inherit()
SonObj:Foo()
print(IsSub(clsSon, clsParent))
print(IsSub(clsSon, clsObject))
以上所述就是本文的全部?jī)?nèi)容了,希望能夠?qū)Υ蠹沂炀氄莆誰ua腳本有所幫助。
您可能感興趣的文章:- Lua中類的實(shí)現(xiàn)原理探討(Lua中實(shí)現(xiàn)類的方法)
- Lua實(shí)現(xiàn)類繼承