Lua中的环境概念
【前言】
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中的环境概念的更多相关文章
- lua中的闭包概念的学习笔记
1.闭包的由来: 个人理解,lua中之所以出现闭包的概念,完全是因为lua中允许函数的嵌套定义,并且在内嵌函数中使用了外包函数中定义的局部变量,例如c.c#就不允许函数的嵌套定义(但是允许函数的嵌套调 ...
- Lua中的模块与包
[前言] 从Lua5.1版本开始,就对模块和包添加了新的支持,可是使用require和module来定义和使用模块和包.require用于使用模块,module用于创建模块.简单的说,一个模块就是一个 ...
- PowerShell_零基础自学课程_5_自定义PowerShell环境及Powershell中的基本概念
PowerShell_零基础自学课程_5_自定义PowerShell环境及Powershell中的基本概念 据我个人所知,windows下的cmd shell除了能够通过修改系统参数来对其中的环境变量 ...
- Lua中的weak表——weak table
弱表(weak table)是一个很有意思的东西,像C++/Java等语言是没有的.弱表的定义是:A weak table is a table whose elements are weak ref ...
- Lua中的require
lua中的require机制 为了方便代码管理,通常会把lua代码分成不同的模块,然后在通过require函数把它们加载进来.现在看看lua的require的处理流程.1.require机制相关 ...
- [译] Closures in Lua - Lua中的闭包
原文:(PDF) . 摘要 一等(first-class)函数是一种非常强大的语言结构,并且是函数式语言的基础特性.少数过程式语言由于其基于栈的实现,也支持一等函数.本文讨论了Lua 5.x用于实现一 ...
- 【转载】lua中的require机制
[转载自]http://blog.chinaunix.net/uid-552961-id-2736410.html lua中的require机制 为了方便代码管理,通常会把lua代码分成不同的模块,然 ...
- Lua中的weak表——weak table(转)
弱表(weak table)是一个很有意思的东西,像C++/Java等语言是没有的.弱表的定义是:A weak table is a table whose elements are weak ref ...
- Lua中的require(转)
lua中的require机制 为了方便代码管理,通常会把lua代码分成不同的模块,然后在通过require函数把它们加载进来.现在看看lua的require的处理流程.1.require机制相关 ...
随机推荐
- Python变量的本质与intern机制
变量的存储 a = 'abc' 理解:①先在内存中生成一个字符串‘abc’ ②可以把比变量名a看做一个便利贴,然后将a贴到‘abc’中 ③注意顺序,是生成‘abc’,然后再创建a指向‘abc’ ...
- mysql 解压版安装
1.官网下载压缩包 2.解压 3.配置环境变量 添加系统环境变量 MYSQL_HOME 值为解压的主目录,例如 D:\mysql-5.7.25-winx64 修改Path 环境变量,点击编辑 ...
- mysql 5.7 json
项目中使用的mysql5.6数据库,数据库表一张表中存的字段为blob类型的json串数据.性能压测中涉及该json串处理效率比较低,开发人员提到mysql5.7版本后json串提供了原生态的json ...
- CodeSmith如何生成实体类
CodeSmith如何生成实体类 这是模板,然后选择对应的表.就可以生成 了 <%-- Name: Database Table Properties Author: Paul Welter D ...
- 近期学习docker遇到的一些问题
最近看某谷的springboot视频,看到了docker部分,在实践过程中遇到了一些问题 默认国外镜像,下载软件很慢 linux内核版本过低,与docker运行不匹配 这里记录一下解决方案 第一个问题 ...
- CODEVS 3546 矩阵链乘法
http://codevs.cn/problem/3546/ 题目 给定有n个要相乘的矩阵构成的序列(链)<A1,A2,A3,.......,An>,要计算乘积A1A2.....An.一组 ...
- java 11 新的Epsilon垃圾收集器
A NoOp Garbage Collector JDK上对这个特性的描述是: 开发一个处理内存分配但不实现任何实际内存回收机制的GC, 一旦可用堆内存用完, JVM就会退出. 如果有System.g ...
- nginx安装访问
依赖包安装: 安装gcc gcc-c++: yum -y install gcc gcc-c++ autoconf automake 安装pcre: yum -y install pcre pcre- ...
- vue实战记录(六)- vue实现购物车功能之地址列表选配
vue实战,一步步实现vue购物车功能的过程记录,课程与素材来自慕课网,自己搭建了express本地服务器来请求数据 作者:狐狸家的鱼 本文链接:vue实战-实现购物车功能(六) GitHub:sue ...
- filebeat+logstash配置
一. filebeat.yml的配置 filebeat.prospectors:- input_type: log paths: - /tmp/logs/optimus-activity-api.lo ...