Lua中全局變量與非全局環(huán)境介紹
今天來聊兩個(gè)話題——全局變量和非全局環(huán)境。
正如大家目前心里所感受到的,全局變量的內(nèi)容很簡(jiǎn)單,而非全局環(huán)境的內(nèi)容就稍微要鍛煉一下腦細(xì)胞了。
1.全局變量的原形
在Lua中,要聲明全局變量很簡(jiǎn)單,那就是定義變量的時(shí)候,前面不要加上local。
這個(gè)神秘的全局變量,其實(shí)本質(zhì)上也是一個(gè)table,它把我們創(chuàng)建的全局變量都保存到一個(gè)table里了。
而這個(gè)table的名字是:_G
我們來看看代碼:
-- 定義一個(gè)全局變量
gName = "哎喲,很挫哦";
-- 用三種方式輸出變量的值
print(gName);
print(_G["gName"]);
print(_G.gName);
輸出結(jié)果如下:
[LUA-print] 哎喲,很挫哦
[LUA-print] 哎喲,很挫哦
[LUA-print] 哎喲,很挫哦
我們定義了一個(gè)全局變量gName,于是這個(gè)gName成為了_G的一個(gè)字段。
怎么樣,很簡(jiǎn)單吧。
2.非全局的環(huán)境
對(duì)于全局變量,不管到了哪個(gè)地方,哪種語言,大家總是會(huì)告誡說:“不要濫用,后果自負(fù)”
也許是因?yàn)檫@樣,所以Lua有了一種比較特殊的機(jī)制:非全局環(huán)境。
我稱它為“不會(huì)造成全局影響的全局變量”。
3.改變函數(shù)的全局變量環(huán)境——setfenv函數(shù)
先看看以下代碼:
-- 定義一個(gè)全局變量
gName = "哎喲,很挫哦";
-- 將當(dāng)前全局環(huán)境重新設(shè)置為新的table
setfenv(1, {});
-- 輸出值
print(gName);
如果現(xiàn)在運(yùn)行代碼,輸出結(jié)果將會(huì)是這樣的:
[LUA-print] LUA ERROR: [string "src/main.lua"]:107: attempt to call global ‘print' (a nil value)
為什么?很出乎意料的臉print函數(shù)都無法找到了?
這是因?yàn)槲覀円呀?jīng)把當(dāng)前函數(shù)范圍內(nèi)的全局變量環(huán)境改變了,全局變量默認(rèn)是保存在_G中的,而現(xiàn)在的全局變量是在一個(gè)新的table里。
目前這個(gè)table是空的,所以不存在任何全局變量。
setfenv函數(shù)就是用來改變某個(gè)函數(shù)范圍里的全局環(huán)境的,通俗地說,就是把某個(gè)函數(shù)范圍內(nèi)的_G給弄沒了。
setfenv函數(shù)兩個(gè)參數(shù)分別代表:
1). 第一個(gè)參數(shù),可以是即將要改變環(huán)境的函數(shù),也可以是一個(gè)數(shù)字。數(shù)字1代表當(dāng)前函數(shù),數(shù)字2代表調(diào)用當(dāng)前函數(shù)的函數(shù),后面以此類推。
2).第二個(gè)參數(shù),新的全局環(huán)境table。
4.保留原來的_G
現(xiàn)在連print函數(shù)都無法使用了,對(duì)于測(cè)試很不方便,我們可以做個(gè)小動(dòng)作,把原來的_G保留起來。
如下代碼:
-- 定義一個(gè)全局變量
gName = "哎喲,很挫哦";
-- 將當(dāng)前全局環(huán)境重新設(shè)置為新的table
setfenv(1, {g = _G});
-- 輸出值
g.print(gName);
-- 再次定義一個(gè)全局變量
gName = "哎喲,有點(diǎn)錯(cuò)哦";
-- 再次輸出值
g.print(gName);
-- 輸出原來的值
g.print(g.gName);
只要在定義新的環(huán)境時(shí),把_G作為一個(gè)字段放到新的table里,就可以調(diào)用原來的全局變量了。
那么,輸出結(jié)果如下:
[LUA-print] nil
[LUA-print] 哎喲,有點(diǎn)錯(cuò)哦
[LUA-print] 哎喲,很挫哦
三次調(diào)用g.print函數(shù)的輸出結(jié)果都是不一樣的:
a.第一次,此時(shí)剛剛重新設(shè)置了全局環(huán)境,這時(shí)候當(dāng)前函數(shù)的全局變量只有一個(gè),那就是g,所以gName的值是nil。
b.第二次,我們?cè)僖淮螌?duì)gName進(jìn)行賦值,此時(shí),已經(jīng)在新的環(huán)境中了,所以接下來輸出的gName值是存在的。
c.第三次,這次輸出的是g.gName的值,通過g調(diào)用的gName值是原先的全局環(huán)境里的值,所以gName的值仍然是最初的“哎喲,很挫哦”。
其實(shí),這有什么用呢?倒不如直接用局部變量好了。
確實(shí),從這例子里看不出什么特別的地方。
書里對(duì)于知識(shí)的介紹都是由淺入深的,所以這里暫時(shí)也沒有更深入的介紹,看到后面內(nèi)容的時(shí)候,我再繼續(xù)和大家分享。
5.使用__index元方法保留原來的_G
這里還有一個(gè)小技巧分享一下,剛剛舉例保留_G,但是調(diào)用print等函數(shù)時(shí)還需要形如g.print的方式,有點(diǎn)礙事。
我們可以利用__index來解決這個(gè)問題,如下代碼:
-- 定義一個(gè)全局變量
gName = "哎喲,很挫哦";
-- 一個(gè)table,即將成為新的環(huán)境
local newG = {};
setmetatable(newG, {__index = _G});
-- 將當(dāng)前全局環(huán)境重新設(shè)置為新的table
setfenv(1, newG);
gName = "別再哎喲了,很煩!";
-- 輸出值
print(gName);
print(_G.gName);
我們給新的table設(shè)置一個(gè)元表,這個(gè)元表的__index元方法就是_G。
于是,當(dāng)新的環(huán)境里找不到print字段時(shí),就會(huì)去_G里尋找。
輸出結(jié)果如下:
[LUA-print] 別再哎喲了,很煩!
[LUA-print] 哎喲,很挫哦
第一次輸出的是新環(huán)境里的gName值,第二次輸出的是原來環(huán)境里的gName值,互不影響。
6.結(jié)束
好了,關(guān)于全局變量和非全局環(huán)境,就暫時(shí)說這么多。
雖然暫時(shí)還感覺不到有什么作用,沒關(guān)系,后面還會(huì)有關(guān)于這部分的內(nèi)容。
就像__index一樣,是基礎(chǔ),后面可能會(huì)經(jīng)常提到。
相關(guān)文章
Lua編程示例(三):稀疏表、雙端隊(duì)列、格式化輸出、表和循環(huán)表的格式化輸出
這篇文章主要介紹了Lua編程示例(三):稀疏表、雙端隊(duì)列、格式化輸出、表和循環(huán)表的格式化輸出,本文直接給出實(shí)例代碼,代碼中包含詳細(xì)注釋,需要的朋友可以參考下2015-07-07舉例簡(jiǎn)介L(zhǎng)ua中函數(shù)的基本用法
這篇文章主要介紹了舉例簡(jiǎn)介L(zhǎng)ua中函數(shù)的基本用法,--兩個(gè)橫線開始單行的注釋,--[[加上兩個(gè)[和]表示多行的注釋--]],需要的朋友可以參考下2015-07-07Lua獲取網(wǎng)絡(luò)時(shí)間(獲取時(shí)間同步服務(wù)器的時(shí)間)
這篇文章主要介紹了Lua獲取網(wǎng)絡(luò)時(shí)間(獲取時(shí)間同步服務(wù)器的時(shí)間),本文使用Lua作為客戶端獲取網(wǎng)絡(luò)上的一些授時(shí)服務(wù)提供商的時(shí)間,需要的朋友可以參考下2015-04-04