# 从架构层面杜绝lua中使用未定义的变量
标签(空格分隔): lua
---
lua中有一个很坑的地方: 1.就是如果一个变量拼写错误,会自动的认为你定义了一个值为nil的全局变量. 2.如果在function中定义一个局部变量,忘记了local,则会默认为是定义成为了一个全局变量.
为了防止这种问题,我今天想到一个解决方案. 可以达到以下效果: 1.在function中使用一个未定义的变量时则报错. 2.在function中定义全局变量则报错,即全局变量中能够在全局空间中定义.
具体做法如下: 在代码初始化的时候,重写_G的__newindex,这个是在往_G中加新东西的时候会调用的,即定义全局变量的时候会调用. 首先我们用一个table来记录定义了那些全局变量,这样做的目的是,我们之后可以在使用全局变量的时候判断时候已经定义了该全局变量.所以我们还应该重写__index,这个__index是在使用表中的内容会调用的. 所以有了如下代码: ``` local tbDefineValue = {}
setmetatable(_G, { __index = function(tb, key) if not tbDefineValue[key] then error(string.format("找不到全局变量:%s", key)) return nil end rawget(tab, key) end, _newindex = function(tb, key, value) tbDefineValue[key] = true rawset(tb, key, value) end }
) ``` 我们把这段代码放到一个全局空间的最先require的文件里面,做初始化. 这样之后我们每定义一个全局对象都会记录. 另外,很关键的一点是:这个代码会在初始化的调用.所以之后定义的全局变量会记录,使用全局变量会检查时候有该全局变量,如果有就返回,否则报错;
然后在程序运行过程中,我们需要有一个init函数来处理. 在运行过程中,我们需要禁止创建全局变量,只能够改变现有全局变量的值或者使用现有的全局变量. 所以我们需要禁止_G的__newindex的时候去增加全局变量. 所以我们有一个```init()```函数,用于设置_G只读. 代码如下: ``` function init() setmetatable(_G, { __newindex = function(tb, key, value) do return end -- tbDefineValue[key] = true -- rawset(tb, key, value) end, }) end ``` 大功告成. 以下是测试用例. ``` local tbDefineValue = {} setmetatable(_G, { __index = function(tb, key) if not tbDefineValue[key] then error(string.format("can not find global:%s", key)) return nil end rawget(tab, key) end, _newindex = function(tb, key, value) tbDefineValue[key] = true rawset(tb, key, value) end } )
function disableWrite_G() setmetatable(_G, { __newindex = function(tb, key, value) error(string.format("can not define global value:%s in local space", key)) do return end end, }) end
c = 12 print(c)
function test2() d = 12 c = 13 end
function test3() print(c) if a == 12 then print(123) end end
function start() disableWrite_G() test2() test3() end
start() ``` 输出结果是 ``` D:\Binaries-LuaDist-batteries-0.9.8-Windows-x86\bin\lua.exe: .\test.lua:20: can not define global value锛歞 in local space stack traceback: [C]: in function 'error' .\test.lua:20: in function <.\test.lua:19> .\test.lua:30: in function 'test2' .\test.lua:43: in function 'start' .\test.lua:47: in main chunk [C]: ?
```
|
请发表评论