【前言】

Lua将其所有的全局变量保存在一个常规的table中,这个table称为“环境”。这种组织结构的优点在于,其一,不需要再为全局变量创造一种新的数据结构,因此简化了Lua的内部实现;另一个优点是,可以像其他table一样操作这个table。为了便于实施这种操作,Lua将环境table自身保存在一个全局变量_G中。例如,我们可以使用以下代码打印当前环境中所有全局变量的名称。

for n in pairs(_G) do print(n) end

在你的电脑上运行一下以上代码,看看结果。

【全局变量声明】

在Lua中,全局变量不需要声明就可以直接使用,但是这样违反了编程的大忌,随便使用全局变量,将导致程序的性能,当出现bug时,也很难去发现,同时也污染了程序中的命名。考虑到全局变量也是存放在一个table中,我们则可以通过元表来改变其它代码访问全局变量时的行为,看到了么?又是元表。代码如下:

setmetatable(_G, {
__newindex = function (_, k)
error("Attempt to write to undeclared variable " .. k)
end,
__index = function (_, k)
error("Attempt to read undeclared variable " .. k)
end
}) print(a) -- 这里a就是一个全局变量

而有的时候,我们的确需要定义一个全局变量,那怎么办?还记得我在《Lua中的元表与元方法》这篇文章中写的吗?使用rawset就可以完成,它是不同过元表的,直接设置table的值;同时,为了测试一个变量是否存在,就不能简单的将它与nil比较。因为如果它为nil,访问就会抛出一个错误,同样,我们可以使用rawget来绕过元方法。

【非全局的变量】

由于“环境”这个概念是全局的,任何对他的修改都会影响程序的所有部分。例如:若安装一个元表用于控制全局变量的访问,那么整个程序都必须遵循这个规范。但使用某个库时,没有先声明就使用了全局变量,那么这个程序就无法运行了。

可以通过函数setfenv来改变一个函数的环境。该函数的参数是一个函数和一个新的环境table。第一个参数除了可以指定为函数本身,还可以指定为一个数字,以表示当前函数调用栈中的层数。数字1表示当前函数,数字2表示调用当前函数的函数,以此类推。首先来一小段代码:

a =  -- 这里创建了一个全局变量

-- 将当前环境变量改为一个新的空table
setfenv(, {})
print(a)

运行代码会弹出这样的错误:attempt to call global ‘print’ (a nil value)

print是存放在_G中的,由于我们将当前的环境变量重置为了一个空的table,导致找不到print了,所以就出现了错误。为了防止这样的错误的放生,在我们改变当前的环境变量之前,我们需要保存当前的环境变量。看下面的代码:

a =  -- 这里创建了一个全局变量

-- 将当前环境变量改为一个新的空table
setfenv(, {g = _G})
g.print(a) -- 输出nil
g.print(g.a) -- 输出1

这个时候访问g就会得到原来的环境,这个环境中包含了字段print。我们可以使用名字_G来代替g,如下述代码:

a =  -- 这里创建了一个全局变量

-- 将当前环境变量改为一个新的空table
setfenv(, {_G = _G})
_G.print(a) -- 输出nil
_G.print(_G.a) -- 输出1

不要忘了我们之前总结的__index元方法,我们可以设置新的环境变量的__index为_G,这样,当在新的环境中找不到对应的变量时,就会去_G中找,这样,就相当于新的环境变量继承了全局的环境变量_G,看以下代码:

a =  -- 这里创建了一个全局变量

local newEnv = {}
setmetatable(newEnv, {__index = _G}) -- 将当前环境变量改为一个新的空table
setfenv(, newEnv)
print(a)

在Lua中,函数会继承创建其的环境,所以一个程序块若改变了它自己的环境,那么后续由它创建的函数都将共享这个新环境。这项机制对于创建名称空间是很有用的。之后的总结中还会继续讲解的。

Lua中的环境概念的更多相关文章

  1. lua中的闭包概念的学习笔记

    1.闭包的由来: 个人理解,lua中之所以出现闭包的概念,完全是因为lua中允许函数的嵌套定义,并且在内嵌函数中使用了外包函数中定义的局部变量,例如c.c#就不允许函数的嵌套定义(但是允许函数的嵌套调 ...

  2. Lua中的模块与包

    [前言] 从Lua5.1版本开始,就对模块和包添加了新的支持,可是使用require和module来定义和使用模块和包.require用于使用模块,module用于创建模块.简单的说,一个模块就是一个 ...

  3. PowerShell_零基础自学课程_5_自定义PowerShell环境及Powershell中的基本概念

    PowerShell_零基础自学课程_5_自定义PowerShell环境及Powershell中的基本概念 据我个人所知,windows下的cmd shell除了能够通过修改系统参数来对其中的环境变量 ...

  4. Lua中的weak表——weak table

    弱表(weak table)是一个很有意思的东西,像C++/Java等语言是没有的.弱表的定义是:A weak table is a table whose elements are weak ref ...

  5. Lua中的require

    lua中的require机制    为了方便代码管理,通常会把lua代码分成不同的模块,然后在通过require函数把它们加载进来.现在看看lua的require的处理流程.1.require机制相关 ...

  6. [译] Closures in Lua - Lua中的闭包

    原文:(PDF) . 摘要 一等(first-class)函数是一种非常强大的语言结构,并且是函数式语言的基础特性.少数过程式语言由于其基于栈的实现,也支持一等函数.本文讨论了Lua 5.x用于实现一 ...

  7. 【转载】lua中的require机制

    [转载自]http://blog.chinaunix.net/uid-552961-id-2736410.html lua中的require机制 为了方便代码管理,通常会把lua代码分成不同的模块,然 ...

  8. Lua中的weak表——weak table(转)

    弱表(weak table)是一个很有意思的东西,像C++/Java等语言是没有的.弱表的定义是:A weak table is a table whose elements are weak ref ...

  9. Lua中的require(转)

    lua中的require机制    为了方便代码管理,通常会把lua代码分成不同的模块,然后在通过require函数把它们加载进来.现在看看lua的require的处理流程.1.require机制相关 ...

随机推荐

  1. cpu_ops、suspend_ops、arm_idle_driver以及machine_restart/machine_power_off到底层PSCI Firmware分析

    在内核中针对的cpu的操作,比如arm_cpuidle_init.arm_cpuidle_suspend.boot_secondary.secondary_start_kernel.op_cpu_di ...

  2. 证明与计算(4): 完美散列函数(Perfect Hash function)

    原文:wiki: 完美散列函数 假设,写一个SQL语句解析器,词法分析对SQL语句解析,把语句分成了多个token,一般这个时候会需要查询这个token是否是一个关键字token. 例如keyword ...

  3. 第四章· Redis的事务、锁及管理命令

    一.事务介绍 二.Redis乐观锁介绍 三.Redis管理命令 一.事务介绍 Redis的事务与关系型数据库中的事务区别 1)在MySQL中讲过的事务,具有A.C.I.D四个特性 Atomic(原子性 ...

  4. Linux(Ubuntu)使用日记(三)------git安装使用

    1. 安装 首先,确认你的系统是否已安装git,可以通过git指令进行查看,如果没有,在命令行模式下输入sudo apt-get install git命令进行安装. 2.  配置 git confi ...

  5. python中的编码问题

    遇到的问题: (1)ValueError: source code string cannot contain null bytes 发现文件的编码被改成了UTF-16BE,使用pycharm设置项目 ...

  6. Python——迭代器

    一.概述 迭代器是访问集合元素的一种方式.迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束.迭代器只能往前不会后退. 二.可迭代的对象 序列:字符串.列表.元组 非序列:字典.文件 三 ...

  7. Python——Django-settings.py的内容

    一.HTML路径设置 #所有和HTML路径相关的设置都在这里 TEMPLATES = [ { 'BACKEND': 'django.template.backends.django.DjangoTem ...

  8. hdu-3416(最短路+网络流)

    题意:给你一个有向权图,问你从S到E有几条最短路,每条边直走一次的情况下: 解题思路:每条边直走一次,最大流边权为1,因为要算几条最短路,那么能得到最短路的路径标记下,然后跑最大流 代码: #incl ...

  9. Django 序列化

    序列化 背景 对于Django 的queryset 对象在传递给 前端的时候,前端是无法识别的 因此需要存在一个转换过程将 queryset 对象转换成 字符串前端才可以识别 演示 QuerySet ...

  10. Scrapy 框架 中间件,信号,定制命令

    中间件 下载器中间件 写中间件 from scrapy.http import HtmlResponse from scrapy.http import Request class Md1(objec ...