环境问题

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)()

例如,当 varnamex 时,那么上面字符串连接的结果就是 "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,就可以完成这项任务,并在路径中的中间表不存在时,创建出他们。

图 22.1 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

(其中带 falseor,确保了这个新全局变量,会始终得到一个与 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:“检查全局变量声明”

图 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。所谓 自由名字 ,是个未与显式声明绑定的名称,也就是说,他不会出现在某个对应的局部变量作用域中。例如,在下面的代码块中,xy 都是自由名字,但 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 变量。

  • 1该机制是 Lua 从 5.1 版到 5.2 版变化最大的部分之一。以下讨论几乎不适用于 Lua 5.1。

  • 2老实说,Lua 在错误信息中就使用了这个名字,因此他会将涉及变量 _ENV.x 的错误,报告为有关 global x 的错误。

使用 _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

在这段代码中,新环境继承了全局环境中的 printa。不过,任何赋值都会进到新表。通过 _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

这种技术需要更多工作,但他能更好地记录模组依赖。

_ENVload

正如我(作者)之前提到的,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?sinstring!!!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)

Last change: 2025-01-10, commit: 3638671

小额打赏,赞助 xfoss.com 长存......

微信 | 支付宝

若这里内容有帮助到你,请选择上述方式向 xfoss.com 捐赠。