1. 程式人生 > >Lua中調用函數使用點號和冒號的區別

Lua中調用函數使用點號和冒號的區別

別了 tab tle sel n) self 方式 fun 輸出

1.初學者最易混亂Top1——調用函數時用點號還是用冒號?

我們來看看下面的兩句代碼:

mSprite.setPosition(100, 20);
mSprite:setPosition(100, 20);

對於初次接觸Lua的朋友來說,這簡直就是噩夢,為嘛函數的調用有兩種形式,是讓我們隨便挑的意思嗎?

這兩種形式是有區別的,區別很大,但只有一個。
不過,暫時不解釋,後面再介紹。

2.最簡單的類

我們先來看看簡單的,來創建一個“類”試試,如下代碼:

TSprite = {
        x = 0,
        y = 0,
    }
    function TSprite.setPosition(x, y)
        TSprite.x 
= x; TSprite.y = y; end TSprite.setPosition(1, 2); print("TSprite坐標(" .. TSprite.x .. "," .. TSprite.y .. ")");

其實就是創建了一個table,給這個table添加一些字段而已。
輸出結果如下:

[LUA-print] TSprite坐標(1,2)


大家留意一下setPosition函數,函數裏其實也是通過TSprite來調用x和y字段的。
並且,我們使用setPosition的方式是,使用點號,這是正宗的函數調用方式,記住了。

3.不用真實姓名可以嗎?——self的作用

如果大家比較敏感的話,就會發現,剛剛的例子很有問題,如果我們這樣調用的話:

   local who = TSprite;
    TSprite = nil;
   
    who.setPosition(1, 2);

這麽做一定會報錯,雖然通過who確實可以成功調用setPosition函數,但函數裏需要用到TSprite,而此時的TSprite已經為nil了。

於是,聰明的我們可以這麽做:

TSprite = {
        x = 0,
        y = 0,
    }
    function TSprite.setPosition(self, x, y)
        self.x 
= x; self.y = y; end local who = TSprite; TSprite = nil; who.setPosition(who, 1, 2); print("TSprite坐標(" .. who.x .. "," .. who.y .. ")");

輸出結果仍然是:

[LUA-print] TSprite坐標(1,2)


留意setPosition的第一個參數,我們強制要求傳入一個參數,這個參數就是TSprite本身。

於是,在調用setPosition函數時,傳入who,who的內容就是TSprite的內容,於是,setPosition就能正常執行了。

4.發揮偷懶的傳統美德——默認的self參數,以及默認傳遞self參數

如果你讓一個高(chao)智(ji)商(lan)猿人每次創建函數和調用函數都要這麽去處理self,那他一定會說“你過來一下下,我保證不打死你”。

所以,Lua提供了一個新的使用方式,沒錯,那就是冒號。

看好了,我說的是,用冒號調用函數。
如下代碼:

 TSprite = {
        x = 0,
        y = 0,
    }
    function TSprite:setPosition(x, y)
        self.x = x;
        self.y = y;
    end
    local who = TSprite;
    TSprite = nil;
     
    who:setPosition(1, 2);

第一,留意setPosition函數的定義,使用了冒號;
第二,留意setPosition函數的調用,使用了冒號。

冒號的作用就是:定義函數時,給函數的添加隱藏的第一個參數self;調用函數時,默認把當前調用者作為第一個參數傳遞進去。

使用了冒號之後,就相當於我們剛剛使用點號時一樣,只是我們不再需要顯式地定義self參數以及主動地傳遞who參數。

好了,這就是點號和冒號的區別了,可以說,冒號是為了給我們偷懶而誕生的。
如果是使用Cocos2d-x lua來開發的話,大部分情況下都是使用冒號的。
原因很簡單,因為大部分情況下我們都要使用到self參數,就像C++的this關鍵字一樣。

Lua中調用函數使用點號和冒號的區別