1

我正在尝试控制交互环境。这是我的尝试:

home: lua
Lua 5.0.3  Copyright (C) 1994-2006 Tecgraf, PUC-Rio
> for n in pairs(_G) do io.write(n) end
stringxpcallLUA_PATHtostringgcinfoloadlibosunpackrequiregetfenvsetmetatablenext_TRACEBACKasserttonumberiorawequalcollectgarbagegetmetatable_LOADEDrawsetmathLUA_CPATHpcalldebug__powtypetablecoroutineprint_Gnewproxyrawgetloadstring_VERSIONdofilesetfenvpairsipairserrorloadfile> 
> G=_G
> _G={}
> setmetatable(_G,{__index=G,__newindex=function() print("nope") end})
> for n in pairs(_G) do io.write(n) end
> x=3
> 

我希望在命令“x=3”之后看到“nope”。然而,它通过了。_G 不是空的吗,_G 的任何访问/更新都将通过定义的元方法?

4

1 回答 1

2

来自 Lua 5.0 手册:

_G

保存全局环境(即 )的全局变量(不是函数_G._G = _G)。Lua 本身不使用这个变量;改变它的值不会影响任何环境。( setfenv用于改变环境。)

更具体地说,在交互式 Lua 解释器中,您需要setfenv( 0, new_global_env )将当前运行线程的环境更改为new_global_env.

例子:

Lua 5.0.3  Copyright (C) 1994-2006 Tecgraf, PUC-Rio
> do
>> local rawget, print, _G = rawget, print, _G
>> local new_global_env = setmetatable( { tostring = tostring }, {
>>   __index = function( t, k )
>>     print( "get", k )
>>     return rawget( _G, k )
>>   end
>> } )
>> setfenv( 0, new_global_env )
>> end
> print( "hello" )
get print
hello
> 

rawgetprint_G的上值以及新的tostring全局值对于避免无限递归是必要的。)

于 2015-11-29T03:29:28.430 回答