博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Step By Step(Lua环境)
阅读量:4110 次
发布时间:2019-05-25

本文共 2455 字,大约阅读时间需要 8 分钟。

   Lua将其所有的全局变量保存在一个常规的table中,这个table被称为“环境”。它被保存在全局变量_G中。

    1. 全局变量声明:
    Lua中的全局变量不需要声明就可以使用。尽管很方便,但是一旦出现笔误就会造成难以发现的错误。我们可以通过给_G表加元表的方式来保护全局变量的读取和设置,这样就能降低这种笔误问题的发生几率了。见如下示例代码:

1 --该table用于存储所有已经声明过的全局变量名 2 local declaredNames = {}  3 local mt = { 4     __newindex = function(table,name,value) 5         --先检查新的名字是否已经声明过,如果存在,这直接通过rawset函数设置即可。 6         if not declaredNames[name] then 7             --再检查本次操作是否是在主程序或者C代码中完成的,如果是,就继续设置,否则报错。 8             local w = debug.getinfo(2,"S").what 9             if w ~= "main" and w ~= "C" then10                 error("attempt to write to undeclared variable " .. name)11             end12             --在实际设置之前,更新一下declaredNames表,下次再设置时就无需检查了。13             declaredNames[name] = true14         end15         print("Setting " .. name .. " to " .. value)16         rawset(table,name,value)17     end,18     19     __index = function(_,name)20         if not declaredNames[name] then21             error("attempt to read undeclared variable " .. name)22         else23             return rawget(_,name)24         end25     end26 }    27 setmetatable(_G,mt)28 29 a = 1130 local kk = aa31 32 --输出结果为:33 --[[34 Setting a to 1135 lua: d:/test.lua:21: attempt to read undeclared variable aa36 stack traceback:37         [C]: in function 'error'38         d:/test.lua:21: in function 
39 d:/test.lua:30: in main chunk40 [C]: ?41 --]]

    2. 非全局的环境:
    全局环境存在一个刚性的问题,即它的修改将影响到程序的所有部分。Lua 5为此做了一些改进,新的特征可以支持每个函数拥有自己独立的全局环境,而由该函数创建的closure函数将继承该函数的全局变量表。这里我们可以通过setfenv函数来改变一个函数的环境,该函数接受两个参数,一个是函数名,另一个是新的环境table。第一个参数除了函数名本身,还可以指定为一个数字,以表示当前函数调用栈中的层数。数字1表示当前函数,2表示它的调用函数,以此类推。见如下代码:

1 a = 1 2 setfenv(1,{}) 3 print(a) 4  5 --输出结果为: 6 --[[ 7 lua: d:/test.lua:3: attempt to call global 'print' (a nil value) 8 stack traceback: 9         d:/test.lua:3: in main chunk10         [C]: ?11 --]]

    为什么得到这样的结果呢?因为print和变量a一样,都是全局表中的字段,而新的全局表是空的,所以print调用将会报错。

    为了应对这一副作用,我们可以让原有的全局表_G作为新全局表的内部表,在访问已有全局变量时,可以直接转到_G中的字段,而对于新的全局字段,则保留在新的全局表中。这样即便是函数中的误修改,也不会影响到其他用到全局变量(_G)的地方。见如下代码:

1 a = 1 2 local newgt = {}  --新环境表 3 setmetatable(newgt,{__index = _G}) 4 setfenv(1,newgt) 5 print(a)  --输出1 6  7 a = 10 8 print(a)  --输出10 9 print(_G.a) --输出110 _G.a = 2011 print(a)  --输出10

    最后给出的示例是函数环境变量的继承性。见如下代码:

1 function factory() 2     return function() return a end 3 end 4 a = 3 5 f1 = factory() 6 f2 = factory() 7 print(f1())  --输出3 8 print(f2())  --输出3 9 10 setfenv(f1,{a = 10})11 print(f1())  --输出1012 print(f2())  --输出3

转载地址:http://vmosi.baihongyu.com/

你可能感兴趣的文章
iOS 操作系统被曝无线网络命名bug 导致 iPhone无法连接无线网络
查看>>
好气!等一年,这个iCloud 账户漏洞竟只值$1.8万?!说好的$35万呢???
查看>>
Linux 应用市场易受RCE和供应链攻击,多个0day未修复
查看>>
可蠕虫 DarkRadiation 勒索软件瞄准 Linux 和 Docker 实例
查看>>
Windows DWrite 组件 RCE 漏洞 (CVE-2021-24093) 分析
查看>>
数百万戴尔设备遭 BIOSConnect 代码执行漏洞影响
查看>>
Atlassian 域名被曝一次点击账户接管漏洞 可导致供应链攻击
查看>>
软件物料清单 (SBOM):从透明度理念到代码落地
查看>>
合勤科技称企业防火墙和VPN设备遭复杂攻击
查看>>
Fortinet 防火墙受高危漏洞影响,可遭远程攻击
查看>>
微软称 SolarWinds 黑客还在继续攻击 IT 企业
查看>>
老旧漏洞不修复,西部数据存储设备数据遭擦除
查看>>
微软 Edge bug 导致黑客窃取用户在任意站点的机密信息,颁发2万美元奖金
查看>>
Phoenix Contact 多款工业产品被曝多个高危漏洞
查看>>
什么鬼?我能通过依赖混淆攻击在 Halo 游戏服务器中执行命令,微软不 care?!...
查看>>
谷歌 Compute Engine 的虚拟机曝0day未修复,可遭接管
查看>>
美国特勤局发布十大网络通缉犯名单
查看>>
微软公布Netgear 固件严重漏洞详情,可盗取用户身份并攻陷系统
查看>>
NIST 按行政令关于加强软件供应链安全的要求,给出“关键软件”的定义及所含11类软件...
查看>>
PCB抗干扰设计原则(转)
查看>>