主頁 > 知識庫 > Lua面向?qū)ο笾惡屠^承

Lua面向?qū)ο笾惡屠^承

熱門標簽:上海企業(yè)外呼系統(tǒng)排名 開通400電話申請流程 智能語音電銷的機器人 武漢百應人工智能電銷機器人 電腦外呼系統(tǒng)輻射大嗎 百度地圖標注位置網(wǎng)站 如何利用高德地圖標注家 揚州電銷外呼系統(tǒng)軟件 400手機電話免費辦理

終于來了,在Lua中的面向?qū)ο缶幊蹋嘈拍壳皩W習Lua的大部分人都是為了開發(fā)手機網(wǎng)游吧。

而且基本都是奔著腳本語言的熱更新特性去的,所以全腳本開發(fā)變得十分流行。
對于普及不太廣的Lua(相對于C++、Java等主流語言),需要短時間上手開發(fā)游戲,對新手而言不算簡單。

所以大家才更習慣于繼續(xù)用面向?qū)ο笏枷肴フ垓vLua吧~

好了,不嘮叨了,我最不喜歡嘮叨了。(小若:是是是,你一點都不嘮叨,趕緊開講!)

1.類的對象

至于如何創(chuàng)建一個類,大家已經(jīng)很清楚了,就是一個table而已。
那么,要使用這個類去創(chuàng)建多個對象,又如何實現(xiàn)呢?
使用元表和元方法即可。
 
如下代碼:

復制代碼 代碼如下:

    TSprite = {
        x = 0,
        y = 0,
    }
    function TSprite:setPosition(x, y)
        self.x = x;
        self.y = y;
    end
  
    function TSprite:new()
        o = {}
        setmetatable(o, {__index = self});
        return o;
    end
  
    local who1 = TSprite:new();
    local who2 = TSprite:new();
    who1:setPosition(1, 2);
    who2:setPosition(44, 6);
    print("who1坐標(" .. who1.x .. "," .. who1.y .. ")");
    print("who2坐標(" .. who2.x .. "," .. who2.y .. ")");

留意TSprite的new函數(shù),函數(shù)里創(chuàng)建了一個新的table,并且給新的table設置一個元表,這個元表的__index元方法就是TSprite本身,最后返回這個新的table。

于是,所有通過new生成的新table,都可以使用TSprite的函數(shù)和各個字段屬性(因為__index的值是TSprite)。

因此,我們利用new函數(shù)創(chuàng)建了who1和who2,并且調(diào)用它們的setPosition函數(shù),最后,who1和who2的x、y值都是不同的。
這就是類的對象了。

2.類對象的__index都是同一個TSprite,為什么x、y值可以不相同?

不知道大家有沒有這樣一個疑惑,那就是,為什么who1和who2的x、y是不一樣的,它們最終調(diào)用的不是setPosition函數(shù)么?調(diào)用self.x時最終不是調(diào)用了TSprite的x值么?
這里是會有點混亂,理一理就沒問題了:

1). 當who1里不存在setPosition時,回去__index元方法里查找,于是,會找到TSprite的setPosition函數(shù)
2). 在setPosition函數(shù)里,使用了self.x = x,此時的self就是who1,who1中是不存在x字段的,所以,如果我們要打印self.x的值,則其實是打印了TSprite的x值
3). 但是,注意,但是來了。__index元方法是用于調(diào)用的,而不是用于賦值的,因此,self.x = x這句話,其實只是給who1這個table的x字段賦值了,who1本身不存在x字段,此時給它賦值了,于是who1存在了x字段,以后who1都不會再去TSprite里查找x字段了。
4). 因此,對who1和who2的x、y字段進行賦值操作時,是完全不會影響到TSprite的。

3.節(jié)省資源——使用TSprite作為元表

我們再仔細觀察一下new函數(shù),我們在給新table設置元表的時候,是重新創(chuàng)建了一個元表的:setmetatable(o, {__index = self});

這么做的話,每次調(diào)用new函數(shù)創(chuàng)建一個新對象時,都會產(chǎn)生一個新的元表,雖然這開支似乎可以忽略,但,擁有強迫癥的你,一定很喜歡下面的代碼:

復制代碼 代碼如下:

    function TSprite:new()
        o = {}
        setmetatable(o, self);
        self.__index = self;
        return o;
    end

在這段新的new函數(shù)里,使用self作為元表,然后又使用self作為__index的值。

這么一看,有點繞不過來,我就喜歡大家繞不過來,這樣我又可以嘮叨了:
1). 調(diào)用new函數(shù)時,self其實就是TSprite本身,這里完全可以用TSprite代替,不過,為了給以后做鋪墊,這里還是使用self吧。
2). self.__index = self,不要被這句代碼嚇到了,其實還是那么一回事,設置元表的__index元方法,這里就 相當于TSprite.__index = TSprite。
3). TSprite自己作為__index的值沒問題么?確實沒問題,TSprite也是一個table,table可以作為元表,元表可以有__index元方法,這絲毫沒有英雄。
4). 于是,通過這個小技巧,我們就避免了每次調(diào)用new函數(shù)時都額外創(chuàng)建一個新的元表了。

4.富二代什么的我才不喜歡——繼承

我們總是笑話富二代,但誰的內(nèi)心深處不希望自己是一個富二代呢~
像我這種立志靠自己成為富一代的人,可不多了~(小若:啊我呸~?。?br />  
那么,在Lua里如何實現(xiàn)繼承呢?很簡單,但是需要認真思考,如下代碼:

復制代碼 代碼如下:

    TSprite = {
        x = 0,
        y = 0,
    }
    function TSprite:setPosition(x, y)
        self.x = x;
        self.y = y;
    end
  
    function TSprite:new()
        o = {}
        setmetatable(o, self);
        self.__index = self;
        return o;
    end
  
    local MoneySprite = TSprite:new();
    function MoneySprite:setPosition(x, y)
        print("呵呵,我是富二代,根本不需要改變。");
    end

TSprite仍然沒變,但是,我們看看MoneySprite,按之前的理解,它是TSprite的一個對象。
只是,“對象”這稱呼是我們自己定的,實際上它還是一個table而已。

此時,我們修改了MoneySprite的setPosition函數(shù),于是,調(diào)用MoneySprite的setPosition函數(shù)時,與TSprite無關了。

但,這不是重點,重點是接下來的代碼:

復制代碼 代碼如下:

    local who = MoneySprite:new();
    who:setPosition(44, 6);
  
    print("who坐標(" .. who.x .. "," .. who.y .. ")");

我們再次調(diào)用MoneySprite的new函數(shù)創(chuàng)建了一個新對象。
這又是什么情況呢?關鍵是new函數(shù)里的代碼,此時,new函數(shù)里的self是誰?
new函數(shù)是由MoneySprite調(diào)用的,因此,self就是MoneySprite。
于是新對象的元表就是MoneySprite,元表的__index也是MoneySprite。

因此~!很神奇的,調(diào)用who的setPosition函數(shù)的時候,其實也是調(diào)用了MoneySprite的setPosition函數(shù)。

于是,who就是是MoneySprite的對象,而MoneySprite就是TSprite的子類。

來看看輸出結果吧:

復制代碼 代碼如下:

[LUA-print] 呵呵,我是富二代,根本不需要改變。
[LUA-print] who坐標(0,0)

怎么樣?繼承的實現(xiàn)方法也很簡單吧?
如果對元表、元方法、self比較生疏的話,可能一時間會理解不過來,沒關系,多思考一會,或者隔天再回頭思考,就會豁然開朗了。

5.結束

不知不覺這個系列的文章已經(jīng)寫了20篇了,真是太出乎我的意料了。
我竟然可以堅持下來,但寫文章的效果確實很好,每晚的1個多小時付出也很值得。
起碼,我對Lua基礎的理解又更加鞏固了~
 
好吧,繼續(xù)堅持…(小若:所以說啊~!為什么每次都要用省略號,用感嘆號不是更能表達你的決心嗎…)

您可能感興趣的文章:
  • Lua中的string庫(字符串函數(shù)庫)總結
  • Lua中的函數(shù)(function)、可變參數(shù)、局部函數(shù)、尾遞歸優(yōu)化等實例講解
  • Lua中的一些常用函數(shù)庫實例講解
  • Lua中的模塊與module函數(shù)詳解
  • Lua中的函數(shù)知識總結
  • Lua字符串庫中的幾個重點函數(shù)介紹
  • Lua的table庫函數(shù)insert、remove、concat、sort詳細介紹
  • Lua中的常用函數(shù)庫匯總
  • Lua中的面向?qū)ο缶幊淘斀?/li>
  • Lua面向?qū)ο笾嘀乩^承、私密性詳解
  • Lua面向?qū)ο缶幊虒W習筆記
  • Lua中函數(shù)與面向?qū)ο缶幊痰幕A知識整理

標簽:黑龍江 嘉峪關 張掖 武漢 宜賓 延邊 江西 新余

巨人網(wǎng)絡通訊聲明:本文標題《Lua面向?qū)ο笾惡屠^承》,本文關鍵詞  Lua,面向,對,象之,類和,繼承,;如發(fā)現(xiàn)本文內(nèi)容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Lua面向?qū)ο笾惡屠^承》相關的同類信息!
  • 本頁收集關于Lua面向?qū)ο笾惡屠^承的相關信息資訊供網(wǎng)民參考!
  • 推薦文章