环境问题
The Environment
全局变量,是大多数编程语言的必备之恶,global variables are a necessary evil of most programming languages。一方面,全局变量的使用,很容易导致代码复杂,将程序中显然无关的部分纠缠在一起。另一方面,明智地使用全局变量,可以更好地表达程序中真正的全局内容;此外,全局常量是无害的,但像 Lua 这样的动态语言,却无法从变量中区分出常量来。像 Lua 这样的嵌入式语言,又给这种组合添加了另一成分:全局变量是指在整个程序中可见的变量,但 Lua 没有程序的明确概念,取而代之的是由主机应用程序所调用的代码块(chunks)。
籍由不带全局变量,但却不遗余力地假装自己有全局变量,Lua 解决了这个难题。我们可以近似地认为,Lua 将其所有全局变量,保存在一个称为 全局环境,global environment 的常规表中。在本章的后面部分,我们将看到 Lua 可以将其 “全局” 变量,保存在多个环境中。现在,我们将专注使用第一种近似方法。
使用表来存储全局变量,简化了 Lua 的内部实现,因为全局变量不需要不同的数据结构。另一个优点是,我们可以像操作其他表一样,操作这个表。为了方便操作,Lua 将全局环境本身,存储在全局变量 _G
中。(因此,_G._G
就等于 _G
。)例如,以下代码打印出了定义在全局环境中所有变量的名字:
> for n in pairs(_G) do print(n) end
assert
error
tostring
xpcall
_G
rawlen
warn
coroutine
string
rawequal
collectgarbage
debug
os
load
pcall
type
getmetatable
dofile
loadfile
pairs
Account
setmetatable
_VERSION
a
arg
utf8
math
next
ipairs
package
rawset
print
io
rawget
require
table
tonumber
select
译注:以上是在执行
lua -i lib/account.dual-rep.lua
后,打印出的全局环境中所有变量的名字。下面则是仅执行lua -i
时的情况。
> for n in pairs(_G) do print(n) end
utf8
collectgarbage
string
pairs
require
io
_VERSION
rawget
package
select
math
error
xpcall
setmetatable
dofile
ipairs
load
warn
arg
debug
os
assert
rawlen
tonumber
getmetatable
_G
rawequal
coroutine
loadfile
type
rawset
next
print
table
tostring
pcall
对比二者的差异,可以发现代码块中定义的公开变量会进入到全局环境中,局部变量则不会进入到全局环境。
有着动态名字的全局变量
Global Variables with Dynamic Names
通常,对于访问和设置全局变量来说,赋值操作就足够了。不过,有时我们会需要某种形式的元编程,some form of meta-programming,比如当我们需要操作某个名字存储在另一变量中,或者名字是在运行时,以某种方式计算出来的全局变量时。为了获取这样一个变量的值,有些程序员会这样写:
value = load("return " .. varname)()
例如,当 varname
为 x
时,那么上面字符串连接的结果就是 "return x"
,运行后就能得到想要的结果。不过,这段代码涉及到一个新代码块的创建和编译,开销就会有点高。我们可以用下面的代码,来达成同样的效果,他比上面的代码的效率,要高出不止一个数量级:
value = _G[varname]
由于环境是个常规表,因此我们只需用所需的键(变量名),对其进行索引即可。
以类似的方式,我们可以通过写下 _G[varname] = value
,来为名字为动态计算出的全局变量赋值。但请注意:有些程序员对这些功能感到有点兴奋,而最终编写出类似 _G["a"] = _G["b"]
的代码,这只是 a = b
的一种复杂写法。
上面这个问题的概括,便是允许在动态名字中使用字段,如 "io.read"
或 "a.b.c.d"
。如果我们写下 _G["io.read"]
,显然就无法获得从 io
表中获取到 read
字段。但我们可以编写一个函数 getfield
,使 getfield("io.read")
返回预期的结果。这个函数主要是个循环,从 _G
开始,逐个字段进行推进:
function getfield (f)
local v = _G -- 从全局变量表开始
for w in string.gmatch(f, "[%a_][%w_]*") do
v = v[w]
end
return v
end
我们依靠 gmatch
遍历 f
中的所有标识符。
设置字段的相应函数,就会稍微复杂一些。像 a.b.c.d = v
这样的赋值,相当于下面的代码:
local temp = a.b.c
temp.d = v
也就是说,我们必须检索到最后的名字,然后单独处理这个最后的名字。图 22.1 “函数 setfield
” 中的函数 setfield
,就可以完成这项任务,并在路径中的中间表不存在时,创建出他们。
function setfield (f, v)
local t = _G -- 从全局变量表开始
for w, d in string.gmatch(f, "([%a_][%w_]*)(%.?)") do
if d == "." then -- 不是最后的名字?
t[w] = t[w] or {} -- 在缺失时创建表
t = t[w] -- 获取到该表
else -- 是最后的名字时
t[w] = v -- 进行赋值
end
end
end
这里的模式会捕获变量 w
中的字段名字,以及变量 d
中可选的后跟点。如果字段后面没有点,则其就是最后的名字了。
有了前面的函数,接下来的调用将创建一个全局表 t
、另一个表 t.x
,并将 10
赋值给 t.x.y
:
$ lua -i lib.lua
Lua 5.4.4 Copyright (C) 1994-2022 Lua.org, PUC-Rio
> Lib.setfield("t.x.y", 10)
> t.x.y
10
> Lib.getfield("t.x.y")
10
全局变量的声明
Global-Variable Declarations
Lua 中的全局变量不需要声明。虽然这种行为对小型程序来说很方便,但在稍大的一些程序中,一个简单拼写问题,就可能导致难以发现的错误。不过,如果我们愿意,也可以改变这种行为。由于 Lua 将全局变量保存在常规表中,因此我们可以使用元表,来检测 Lua 于何时访问不存在的变量。
第一种方法仅是检测任何对全局表中不存在键的访问:
Lib.setfield("_PROMPT", "*-*: ")
setmetatable(_G, {
__newindex = function (_, n)
error("尝试写入未声明的变量 " .. n, 2)
end,
__index = function (_, n)
error("尝试读取未声明的变量 " .. n, 2)
end
})
这段代码后,任何访问不存在全局变量的尝试,都会引发错误:
$ lua -i lib.lua
Lua 5.4.4 Copyright (C) 1994-2022 Lua.org, PUC-Rio
*-*: print(a)
stdin:1: 尝试读取未声明的变量 a
stack traceback:
[C]: in function 'error'
lib.lua:208: in metamethod 'index'
stdin:1: in main chunk
[C]: in ?
但是我们如何声明新变量呢?一个选项是使用会绕过元方法的 rawset
:
(其中带 false
的 or
,确保了这个新全局变量,会始终得到一个与 nil
不同的值。)
一种更简单的选项,是将对新全局变量的赋值,限制在函数内部,而允许在代码块的外层进行自由赋值,restrict assignment to new global variables only inside functions, allowing free assignments in the outer level of a chunk。
要检查某个赋值是否在主块中,我们就必须使用调试库。调用 debug.getinfo(2, "S")
会返回一个表,其中的 what
字段,会显示调用元方法的函数,是个主代码块、常规 Lua 函数,还是 C 函数。(我们将在 “内省设施,Introspective Facilities” 小节中,详细介绍 debug.getinfo
)。使用这个函数,我们可以像下面这样重写 __newindex
元方法:
__newindex = function (t, n, v)
local w = debug.getinfo(2, "S").what
if w ~= "main" and w ~= "C" then
error("尝试写入未声明的变量 " .. n, 2)
end
rawset(t, n, v)
end,
这个新版本还接受来自 C 代码的赋值,因为这种代码通常知道他在做什么。
当我们需要测试某个变量是否存在时,我们不能简单地将他与 nil
进行比较,因为如果他为 nil,访问就会引发错误。相反,我们可以使用避免了使用元方法的 rawget
:
if rawget(_G, var) == nil then
-- ‘var’ 未经声明
...
end
目前,咱们的方案不允许全局变量的值为 nil
,因为他们会被自动视为未声明变量。但要解决这个问题并不难。我们只需一个辅助表,来保存那些已声明变量的名字。每当某个元方法被调用时,元方法都会在该表中,检查变量是否为未声明变量。这样的代码可以如下 图 22.2:“检查全局变量声明”。
local declaredNames = {}
setmetatable(_G, {
__newindex = function(t, n, v)
if not declaredNames[n] then
local w = debug.getinfo(2, "S").what
if w ~= "main" and w ~= "C" then
error("尝试写入未经声明的变量"..n, 2)
end
declaredNames[n] = true
end
rawset(t, n, v) -- 执行真正设置
end,
__index = function (_, n)
if not declaredNames[n] then
error("尝试读取未经声明的变量 "..n, 2)
else
return nil
end
end,
})
现在,即使像 x = nil
这样的赋值,也足以声明某个全局变量了。
这两种解决方案的开销都可以忽略不计。在第一种方案下,于正常运行期间,那些元方法永远不会被调用。而在第二种方案下,他们可以被调用,但只有当程序访问某个保存着 nil
的变量时,才会被调用。
非全局的环境
Non-Global Environment
在 Lua 中,全局变量并不需要是真正全局的。正如我(作者)已经暗示过的,Lua 甚至没有全局变量。这乍听起来可能有些奇怪,因为这本书中我们一直在使用全局变量。正如我(作者)曾讲过的,Lua 不遗余力地给程序员,制造全局变量的假象。现在我们就来看看 Lua 是如何制造这种假象的1。
首先,让我们忘掉全局变量。相反,我们将从自由名字的概念开始,instead, we will start with the concept of free names。所谓 自由名字 ,是个未与显式声明绑定的名称,也就是说,他不会出现在某个对应的局部变量作用域中。例如,在下面的代码块中,x
和 y
都是自由名字,但 z
不是:
local z = 10
x = y + z
重要部分来了: Lua 编译器会将代码块中的任何自由名字 x
,转换为 _ENV.x
。因此,上一代码块与下面这个完全等价:
local z = 10
_ENV.x = _ENV.y +z
而这个新的 _ENV
变量是什么呢?
_ENV
不可能是全局变量;我们刚刚说过 Lua 没有全局变量。编译器再度发挥了作用,again, the compiler does the trick。我(作者)已经提到过,Lua 会将任何代码块,都视为匿名函数。实际上,Lua 会将我们的原始代码,编译成如下代码:
local _ENV = some value
return funtion (...)
local z = 10
_ENV.x = _ENV.y + z
end
也就是说,Lua 会在一个名为 _ENV
的预定义上值(一个外部的局部变量,a predefined upvalue, an external local variable)存在的情况下,编译任何的代码块。那么,任何变量都要么是本地变量(当其是个有边界的名字时),要么就是 _ENV
中的一个字段,而 _ENV
本身则是个本地变量(一个上值)。
_ENV
的初始值可以是任何表。(实际上,他不需要是个表;稍后会详细介绍。)任何这种表,都被称作环境。为了保留全局变量的假象,Lua 在内部保留了一个用作全局环境的表。通常,当我们加载某个代码块时,函数 load
就会使用该全局环境,初始化这个预定义的上值。所以,我们原来的代码块,就相当于这个:
local _ENV = the global environment
return function (...)
local z = 10
_ENV.x = _ENV.y + z
end
所有这些安排的结果便是,全局环境中的 x
字段值,为 y
字段的值加上 10
。
乍一看,这似乎是操作全局变量的一种相当复杂的方法。我不会说这是最简单的方法,但他提供了更简单实现方式下,难以达到的灵活性。
在继续之前,我们先总结一下 Lua 中对全局变量的处理:
-
编译器会在任何其编译的代码块之外,创建出一个局部变量
_ENV
; -
编译器会将任何自由名字
var
转换为_ENV.var
; -
函数
load
(或loadfile
)会以 Lua 内部保存的一个常规表的全局环境,初始化代码块的第一个上值。
毕竟,这并不复杂。
有些人会感到困惑,因为他们会试图从这些规则中,推断出额外的魔法。其实并没有什么额外的魔法。尤其是前两条规则,是完全由编译器完成的。除了被编译器预定义之外,_ENV
就是个普通的常规变量。在编译器之外,_ENV
这个名字对 Lua 来说,没有任何特殊含义。2 同样,从 x
到 _ENV.x
的转换,是个简单的语法转换,没有任何隐藏含义。特别是,在转换后,_ENV
将按照标准的可见性规则,指向代码中该处可见的任何 _ENV
变量。
使用 _ENV
在本节中,我们将看到一些对 _ENV
所带来灵活性加以探索的方法。请记住,本节中的大多数示例,都必须作为一个单独代码块来运行。如果我们在交互模式下,逐行输入代码,那么每行都将成为不同的代码块,因此每行都将有个不同的 _ENV
变量。而要将一段代码作为单独代码块运行,我们可以从文件中运行它,或者将其封装在 do--end
块中。
因为 _ENV
是个常规变量,所以我们可以像其他变量一样,对其赋值及读取。_ENV = nil
这个赋值,将使其余代码块中,对全局变量的直接访问无效。这对于控制我们的代码使用哪些变量,会非常有用:
local print, sin = print, math.sin
_ENV = nil
print(13)
print(sin(13))
print(math.cos(13))
脚本:env-demo.lua
$ lua env_demo.lua
13
0.42016703682664
lua: env_demo.lua:6: attempt to index a nil value (upvalue '_ENV')
stack traceback:
env_demo.lua:6: in main chunk
[C]: in ?
对自由名字(某个 “全局变量”)的赋值,都会引发类似的错误。
我们可以明确写出 _ENV
,来绕过某个本地的声明:
a = 13 -- 全局的
local a = 12
print(a) --> 12 (局部的)
print(_ENV.a) --> 13 (全局的)
我们可以使用 _G
,实现同样目的:
a = 13 -- 全局的
local a = 12
print(a) --> 12 (局部的)
print(_G.a) --> 13 (全局的)
通常,_G
和 _ENV
指向的是同一个表,但尽管如此,他们是完全不同的实体。_ENV
是个局部变量,所有对 “全局变量” 的全部访问,实际上都是对他的访问。而 _G
则是个全局变量,没有任何特殊地位。根据定义,_ENV
总是指向当前环境;而 _G
则通常指向全局环境,前提是他是可见的,而且没有人改变过他的值,_G
usually refers to the global environment, provided it is visible and no one changed its value。
_ENV
的主要用途,是更改某段代码所用到的环境。一旦我们更改了环境,那么所有全局的访问,都将使用这个新表:
-- 将当前环境修改未一个新的空表
_ENV = {}
a = 1 -- 在 _ENV 中创建出一个字段
print(a)
--> stdina:4: attempt to call global 'print'(a nil value)
注意:实际输出与上面的原书中的输出有差异。
$ lua env_demo.lua
lua: env_demo.lua:4: attempt to call a nil value (global 'print')
stack traceback:
env_demo.lua:4: in main chunk
[C]: in ?
这可能与 Lua 的版本有关。
若新环境是空的,那么我们就失去了所有全局变量,包括 print
。因此,我们应首先以一些有用的值,比如全局环境产生出他:
a = 15 -- 创建出一个全局变量
_ENV = {g = _G} -- 修改当前环境
a = 1 -- 在 _ENV 里创建出一个字段
g.print(_ENV.a, g.a) --> 1 15
现在,当我们访问那个 “全局的” g
(其存活于 _ENV
,而非全局环境中)时,我们就会得到全局环境,Lua 将在其中,找到 print
函数。
使用用 _G
而非 g
,我们可以像下面这样重写上一示例:
a = 15 -- 创建出一个全局变量
_ENV = {_G = _G} -- 修改当前环境
a = 1 -- 在 _ENV 里创建出一个字段
_G.print(_ENV.a, _G.a) --> 1 15
_G
的唯一特殊状态,发生于 Lua 创建初始全局表,其将其字段 _G
指向其自身时。Lua 并不关心这个变量的当前值。尽管如此,当我们有着一个对全局环境的引用时,我们还是习惯于使用这同样的名字,就像我们在这个重写示例中,所做的那样。
另一产生出咱们新环境的方式,是通过继承:
a = 1
local newgt = {} -- 创建新环境
setmetatable(newgt, {__index = _G})
_ENV = newgt -- 设置他
print(a) --> 1
在这段代码中,新环境继承了全局环境中的 print
和 a
。不过,任何赋值都会进到新表。通过 _G
,我们仍然可以更改全局环境中的变量,而不会有错误修改某个全局环境中变量的危险:
-- 继续前一代码块
a = 10
print(a, _G.a) --> 10 1
_G.a = 20
print(_G.a) --> 20
作为常规变量,_ENV
遵循通常的作用域规则。特别是,在一个代码块中定义的函数,会像访问任何其他外部变量一样,访问 _ENV
:
_ENV = {_G = _G}
local function foo ()
_G.print(a) -- 会被编译作 `_ENV._G.print(_ENV.a)`
end
a = 10
foo() --> 10
_ENV = {_G = _G, a = 20}
foo() --> 20
注:在使用了
_ENV
后,执行for i in _G.pairs(_G) do _G.print(i) end
,将打印出如下内容。
assert
math
require
loadfile
debug
arg
load
os
collectgarbage
pairs
_G
string
coroutine
dofile
table
_VERSION
setmetatable
rawget
utf8
print
rawlen
tostring
package
io
getmetatable
warn
select
rawequal
error
next
pcall
type
rawset
xpcall
tonumber
ipairs
这与不使用
_ENV
时自由名字会进入到_G
不同。或许表明使用_ENV
可以防止_G
被污染。
如果我们定义一个名为 _ENV
的新局部变量,那么对一些自由名字的引用,就会绑定到那个新变量:
a = 2
do
local _ENV = {print = print, a = 14}
print(a) --> 14
end
print(a) --> 2 (会回到最初的 _ENV)
因此,要定义某个有着私有环境的函数,就并不困难了:
function factory (_ENV)
return function () return a end
end
f1 = factory{a = 6}
f2 = factory{a = 7}
print(f1()) --> 6
print(f2()) --> 7
这个 factory
函数,创建返回其自己的 “全局” 变量 a
的简单闭包。在闭包被创建出时,其可见的 _ENV
变量,就是外层 factory
函数的参数 _ENV
;因此,每个闭包都将使用自己的外部变量(作为上值),来访问他的自由名字。
运用一般作用域规则,我们还可以通过其他几种方式操纵环境。例如,我们可以让几个函数共用同一个环境,或者让某个函数改变其与别的函数所共用的环境。
环境与模组
Environment and Modules
在其间咱们讨论到怎样编写模组,名为 “Lua 中编写模组的基本方法” 的小节中,我(作者)曾提到,这些方法的一个缺点,那就是他们都太容易污染全局空间了,比如会在某个私有的声明中,忘记了 local
。环境就为解决这个问题,提供了一种有趣技巧。一旦模块的主代码块有了专属环境,那么不仅他的所有函数会共用这个表,他的所有全局变量也都会进入这个表。我们可以将所有公共函数声明为全局变量,他们将自动进入一个单独表。模块只需将此表,赋值给 _ENV
变量。这样,当我们声明 add
函数时,他就会进到 M.add
:
local M = {}
_ENV = M
function add (c1, c2)
return new(c1.r + c2.r, c1.i + c2.i)
end
此外,我们还可以在无需任何前缀下,调用同一模组中的其他函数。在上面的代码中,add
就从环境中获取到了 new
,即他调用了 M.new
。
这种方法为模组提供了良好支持,程序员几乎不需要做额外的事情。他根本不需要前缀。调用导出的函数,和调用私有函数没有区别。如果程序员忘记了某个 local
,他也不会污染全局命名空间;与之相反,若没有这种方法,私有函数就会变成公开函数。
不过,目前我(作者)还是更喜欢原先的基本方法。他可能还需要更多的事情,但得出的代码会清楚表明其做了什么。我(作者)使用了一种简单方法,来避免错误地创建出全局,即将 nil
赋值给 _ENV
。在那之后,任何对某个全局名字的赋值,都会引发错误。这种方法还有一个额外优点,那就是在旧版本的 Lua 中,无需即可工作。(在 Lua 5.1 中,这种给 _ENV
的赋值不会防止错误,但也不会造成任何损害)。
要访问其他模块,我们可以使用上一节讨论过的方法之一。例如,我们可以声明一个保存了全局环境的局部变量:
local M = {}
local _G = _G
_ENV = nil
随后我们就可以在全局的名字前加上 _G
,在模组的名字前加上 M
了。
一种更严谨的方法,是只将我们需要的函数,或至多需要的模组,声明为局部变量:
-- 模组配置
local M = {}
-- 导入部分:
-- 声明出该模组所需的全部外部内容
local sqrt = math.sqrt
local io = io
-- 从这里之后就不再有外部访问了
_ENV = nil
这种技术需要更多工作,但他能更好地记录模组依赖。
_ENV
与 load
正如我(作者)之前提到的,load
通常会以全局环境,初始化所加载代码块的上值 _ENV
。不过,load
有个允许我们为 _ENV
提供不同初始值的可选第四参数。(函数 loadfile
也有类似参数。)
作为初始示例,假设我们有个定义了程序会用到的几个常量和函数的典型配置文件;他可以是这样的:
-- 文件 'config.lua'
width = 200
height = 300
-- ...
我们可以用以下代码加载他:
env = {}
loadfile("config.lua", "t", env) ()
该配置文件中的所有代码,都将在充当了某种沙盒的空环境 env
中运行。特别是,全部的定义都将进入该环境。该配置文件无法影响其他任何东西,即便在失误下。即使是恶意代码,也无法造成太大破坏。他可以进行拒绝服务(DoS)攻击,浪费掉 CPU 时间和内存,但不会造成其他影响。
有时,我们可能打算多次运行某个代码块,每次都使用不同的环境表。在这种情况下,那个额外加载参数,就没有用了。相反,我们还有另外两个选项。
第一个选项是使用调试库中的函数 debug.setupvalue
。顾名思义,setupvalue
允许我们更改给定函数的任何上值。下一个代码片段,说明了他的用法:
f = load("b = 10; return a")
env = {a = 20}
debug.setupvalue(f, 1, env)
print(f()) --> 20
print(env.b) --> 10
其中 setupvalue
调用的第一个参数是函数,第二个是上值的索引,第三个是该上值的新值。对于这种用法,第二个参数始终为 1
:当某个函数代表了一个代码块时,Lua 会确保他只有一个上值,并且这个上值就是 _ENV
。
这个选项的一个小缺点,是他对调试库的依赖。这个库破坏了一些关于程序的一般假设。例如,debug.setupvalue
就破坏了确保我们无法从其词法范围之外,访问局部变量的 Lua 可见性规则。
以多个不同环境,运行某个代码块的另一个选项,便是在加载代码块时,对其稍加改动。请设想我们在代码块前,添加以下一行:
_ENV = ...;
请记住,Lua 会将任何代码块,都编译为可变参数函数,a variadic function。因此,这个额外代码行,会将传递给该代码块的第一个参数,赋值给给 _ENV
这个变量,进而将该参数设置为环境。下面的代码片段,使用了咱们在 练习 16.1 中实现的 loadwithprefix
函数,说明这个想法:
prefix = "_ENV = ...;"
f = loadwithprefix(prefix, io.lines(filename, "*L"))
-- ...
env1 = {}
f(env1)
env2 = {}
f(env2)
练习
练习 22.1:我们在本章开头定义的函数 getfield
过于宽容,因为他会接受像 math?sin
或 string!!!gsub
这样的 “字段”。请重写他,使其仅接受单个点(.
),作为名字分隔符;
练习 22.2:请详细解释下面的程序中发生了什么,以及他将打印出什么内容;
local foo
do
local _ENV = _ENV
function foo () print(X) end
end
X = 13
_ENV = nil
foo()
X = 0
注:运行该程序的输出为:
$ lua exercise-22.2.lua
13
lua: exercise-22.2.lua:10: attempt to index a nil value (upvalue '_ENV')
stack traceback:
exercise-22.2.lua:10: in main chunk
[C]: in ?
其中
exercise-22.2.lua:10
即为X = 0
。
练习 22.3:请详细解释下面的程序中发生了什么,以及将打印出什么内容。
local print = print
function foo (_ENV, a)
print(a + b)
end
foo({b = 14}, 12)
foo({b = 10}, 1)
(End)