时间:2021-05-22
Lua将其所有的全局变量保存在一个常规的table中,这个table被称为“环境”。它被保存在全局变量_G中。
1. 全局变量声明:
Lua中的全局变量不需要声明就可以使用。尽管很方便,但是一旦出现笔误就会造成难以发现的错误。我们可以通过给_G表加元表的方式来保护全局变量的读取和设置,这样就能降低这种笔误问题的发生几率了。见如下示例代码:
复制代码 代码如下:
--该table用于存储所有已经声明过的全局变量名
local declaredNames = {}
local mt = {
__newindex = function(table,name,value)
--先检查新的名字是否已经声明过,如果存在,这直接通过rawset函数设置即可。
if not declaredNames[name] then
--再检查本次操作是否是在主程序或者C代码中完成的,如果是,就继续设置,否则报错。
local w = debug.getinfo(2,"S").what
if w ~= "main" and w ~= "C" then
error("attempt to write to undeclared variable " .. name)
end
--在实际设置之前,更新一下declaredNames表,下次再设置时就无需检查了。
declaredNames[name] = true
end
print("Setting " .. name .. " to " .. value)
rawset(table,name,value)
end,
__index = function(_,name)
if not declaredNames[name] then
error("attempt to read undeclared variable " .. name)
else
return rawget(_,name)
end
end
}
setmetatable(_G,mt)
a = 11
local kk = aa
--输出结果为:
--[[
Setting a to 11
lua: d:/test.lua:21: attempt to read undeclared variable aa
stack traceback:
[C]: in function 'error'
d:/test.lua:21: in function <d:/test.lua:19>
d:/test.lua:30: in main chunk
[C]: ?
--]]
2. 非全局的环境:
全局环境存在一个刚性的问题,即它的修改将影响到程序的所有部分。Lua 5为此做了一些改进,新的特征可以支持每个函数拥有自己独立的全局环境,而由该函数创建的closure函数将继承该函数的全局变量表。这里我们可以通过setfenv函数来改变一个函数的环境,该函数接受两个参数,一个是函数名,另一个是新的环境table。第一个参数除了函数名本身,还可以指定为一个数字,以表示当前函数调用栈中的层数。数字1表示当前函数,2表示它的调用函数,以此类推。见如下代码:
复制代码 代码如下:
a = 1
setfenv(1,{})
print(a)
--输出结果为:
--[[
lua: d:/test.lua:3: attempt to call global 'print' (a nil value)
stack traceback:
d:/test.lua:3: in main chunk
[C]: ?
--]]
为什么得到这样的结果呢?因为print和变量a一样,都是全局表中的字段,而新的全局表是空的,所以print调用将会报错。
为了应对这一副作用,我们可以让原有的全局表_G作为新全局表的内部表,在访问已有全局变量时,可以直接转到_G中的字段,而对于新的全局字段,则保留在新的全局表中。这样即便是函数中的误修改,也不会影响到其他用到全局变量(_G)的地方。见如下代码:
复制代码 代码如下:
a = 1
local newgt = {} --新环境表
setmetatable(newgt,{__index = _G})
setfenv(1,newgt)
print(a) --输出1
a = 10
print(a) --输出10
print(_G.a) --输出1
_G.a = 20
print(a) --输出10
最后给出的示例是函数环境变量的继承性。见如下代码:
复制代码 代码如下:
function factory()
return function() return a end
end
a = 3
f1 = factory()
f2 = factory()
print(f1()) --输出3
print(f2()) --输出3
setfenv(f1,{a = 10})
print(f1()) --输出10
print(f2()) --输出3
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
今天来聊两个话题——全局变量和非全局环境。正如大家目前心里所感受到的,全局变量的内容很简单,而非全局环境的内容就稍微要锻炼一下脑细胞了。1.全局变量的原形在Lu
1.符号分类(1)全局符号:非静态全局变量,非静态函数(2)外部符号:定义于其它模块,而被本模块引用的全局变量和函数(3)本地符号:静态变量(包括全局和局部),
前言Lua将其所有的全局变量保存在一个常规的table中,这个table称为“环境”。这种组织结构的优点在于,其一,不需要再为全局变量创造一种新的数据结构,因此
变量的作用域和存储方式1.简述变量按作用域的分类变量按作用域分:分为全局变量和局部变量全局变量:在所有函数外部定义的变量叫做全局变量全局变量的使用范围:从定义位
Ruby全局变量全局变量以$开头。未初始化的全局变量的值为nil,在使用-w选项后,会产生警告。给全局变量赋值会改变全局状态,所以不建议使用全局变量。下面的实例