今天來聊兩個話題——全局變量和非全局環(huán)境。
正如大家目前心里所感受到的,全局變量的內容很簡單,而非全局環(huán)境的內容就稍微要鍛煉一下腦細胞了。
1.全局變量的原形
在Lua中,要聲明全局變量很簡單,那就是定義變量的時候,前面不要加上local。
這個神秘的全局變量,其實本質上也是一個table,它把我們創(chuàng)建的全局變量都保存到一個table里了。
而這個table的名字是:_G
我們來看看代碼:
復制代碼 代碼如下:
-- 定義一個全局變量
gName = "哎喲,很挫哦";
-- 用三種方式輸出變量的值
print(gName);
print(_G["gName"]);
print(_G.gName);
輸出結果如下:
復制代碼 代碼如下:
[LUA-print] 哎喲,很挫哦
[LUA-print] 哎喲,很挫哦
[LUA-print] 哎喲,很挫哦
我們定義了一個全局變量gName,于是這個gName成為了_G的一個字段。
怎么樣,很簡單吧。
2.非全局的環(huán)境
對于全局變量,不管到了哪個地方,哪種語言,大家總是會告誡說:“不要濫用,后果自負”
也許是因為這樣,所以Lua有了一種比較特殊的機制:非全局環(huán)境。
我稱它為“不會造成全局影響的全局變量”。
3.改變函數(shù)的全局變量環(huán)境——setfenv函數(shù)
先看看以下代碼:
復制代碼 代碼如下:
-- 定義一個全局變量
gName = "哎喲,很挫哦";
-- 將當前全局環(huán)境重新設置為新的table
setfenv(1, {});
-- 輸出值
print(gName);
如果現(xiàn)在運行代碼,輸出結果將會是這樣的:
復制代碼 代碼如下:
[LUA-print] LUA ERROR: [string "src/main.lua"]:107: attempt to call global ‘print' (a nil value)
為什么?很出乎意料的臉print函數(shù)都無法找到了?
這是因為我們已經把當前函數(shù)范圍內的全局變量環(huán)境改變了,全局變量默認是保存在_G中的,而現(xiàn)在的全局變量是在一個新的table里。
目前這個table是空的,所以不存在任何全局變量。
setfenv函數(shù)就是用來改變某個函數(shù)范圍里的全局環(huán)境的,通俗地說,就是把某個函數(shù)范圍內的_G給弄沒了。
setfenv函數(shù)兩個參數(shù)分別代表:
1). 第一個參數(shù),可以是即將要改變環(huán)境的函數(shù),也可以是一個數(shù)字。數(shù)字1代表當前函數(shù),數(shù)字2代表調用當前函數(shù)的函數(shù),后面以此類推。
2).第二個參數(shù),新的全局環(huán)境table。
4.保留原來的_G
現(xiàn)在連print函數(shù)都無法使用了,對于測試很不方便,我們可以做個小動作,把原來的_G保留起來。
如下代碼:
復制代碼 代碼如下:
-- 定義一個全局變量
gName = "哎喲,很挫哦";
-- 將當前全局環(huán)境重新設置為新的table
setfenv(1, {g = _G});
-- 輸出值
g.print(gName);
-- 再次定義一個全局變量
gName = "哎喲,有點錯哦";
-- 再次輸出值
g.print(gName);
-- 輸出原來的值
g.print(g.gName);
只要在定義新的環(huán)境時,把_G作為一個字段放到新的table里,就可以調用原來的全局變量了。
那么,輸出結果如下:
復制代碼 代碼如下:
[LUA-print] nil
[LUA-print] 哎喲,有點錯哦
[LUA-print] 哎喲,很挫哦
三次調用g.print函數(shù)的輸出結果都是不一樣的:
a.第一次,此時剛剛重新設置了全局環(huán)境,這時候當前函數(shù)的全局變量只有一個,那就是g,所以gName的值是nil。
b.第二次,我們再一次對gName進行賦值,此時,已經在新的環(huán)境中了,所以接下來輸出的gName值是存在的。
c.第三次,這次輸出的是g.gName的值,通過g調用的gName值是原先的全局環(huán)境里的值,所以gName的值仍然是最初的“哎喲,很挫哦”。
其實,這有什么用呢?倒不如直接用局部變量好了。
確實,從這例子里看不出什么特別的地方。
書里對于知識的介紹都是由淺入深的,所以這里暫時也沒有更深入的介紹,看到后面內容的時候,我再繼續(xù)和大家分享。
5.使用__index元方法保留原來的_G
這里還有一個小技巧分享一下,剛剛舉例保留_G,但是調用print等函數(shù)時還需要形如g.print的方式,有點礙事。
我們可以利用__index來解決這個問題,如下代碼:
復制代碼 代碼如下:
-- 定義一個全局變量
gName = "哎喲,很挫哦";
-- 一個table,即將成為新的環(huán)境
local newG = {};
setmetatable(newG, {__index = _G});
-- 將當前全局環(huán)境重新設置為新的table
setfenv(1, newG);
gName = "別再哎喲了,很煩!";
-- 輸出值
print(gName);
print(_G.gName);
我們給新的table設置一個元表,這個元表的__index元方法就是_G。
于是,當新的環(huán)境里找不到print字段時,就會去_G里尋找。
輸出結果如下:
復制代碼 代碼如下:
[LUA-print] 別再哎喲了,很煩!
[LUA-print] 哎喲,很挫哦
第一次輸出的是新環(huán)境里的gName值,第二次輸出的是原來環(huán)境里的gName值,互不影響。
6.結束
好了,關于全局變量和非全局環(huán)境,就暫時說這么多。
雖然暫時還感覺不到有什么作用,沒關系,后面還會有關于這部分的內容。
就像__index一樣,是基礎,后面可能會經常提到。
您可能感興趣的文章:- Lua中的變量類型與語句學習總結
- Lua中的變量和流控制入門學習
- 詳解Lua中的變量相關知識點
- Lua教程(十): 全局變量和非全局的環(huán)境
- Lua判斷變量是否為數(shù)字、字符串是否可以轉換為數(shù)字等
- Lua中創(chuàng)建全局變量的小技巧(禁止未預期的全局變量)
- C語言中通過LUA API訪問LUA腳本變量的簡單例子
- Lua變量類型簡明總結
- Lua中的全局變量、非全局變量總結
- Lua中的變量與賦值方法