码迷,mamicode.com
首页 > 其他好文 > 详细

lua module环境探秘

时间:2016-08-06 01:48:10      阅读:457      评论:0      收藏:0      [点我收藏+]

标签:

module 作用

module (name [, ···])

Creates a module. If there is a table in package.loaded[name], this table is the module. Otherwise, if there is a global table t with the given name, this table is the module. Otherwise creates a new table t and sets it as the value of the global name and the value of package.loaded[name]. This function also initializes t._NAME with the given name, t._M with the module (t itself), and t._PACKAGE with the package name (the full module name minus last component; see below). Finally, module sets t as the new environment of the current function and the new value of package.loaded[name], so that require returns t.

 

http://blog.csdn.net/chenyufei1013/article/details/4438801

当在模块文件中使用module函数的时候,如下所示;

module “mymodule”

。。。

 

实际上等同于以下的语句:

local modname = “mymodule”     – 定义模块名
local M = {}                               -- 定义用于返回的模块表
_G[modname] = M                      -- 将模块表加入到全局变量中
package.loaded[modname] = M    -- 将模块表加入到package.loaded中,防止多次加载
setfenv(1,M)                               -- 将模块表设置为函数的环境表,这使得模块中的所有操作是以在模块表中的,这样定义函数就直接定义在模块表中

。。。

return M

 

 

module被require后返回的对象

整个module文件, 可以看做是一个函数,详见lua chunk解释:

http://www.lua.org/manual/5.1/manual.html

2.4.1 – Chunks

The unit of execution of Lua is called a chunk. A chunk is simply a sequence of statements, which are executed sequentially. Each statement can be optionally followed by a semicolon:

	chunk ::= {stat [`;′]}

There are no empty statements and thus ‘;;‘ is not legal.

Lua handles a chunk as the body of an anonymous function with a variable number of arguments (see §2.5.9). As such, chunks can define local variables, receive arguments, and return values.

 

从上可以看出, module改变了匿名函数的行为, 返回的是 module构造的局部表(env), 此表在module文件中定义, 然后在module声明之后的语句执行前, 被设置为整个匿名函数的环境, 待module声明之后的语句全部执行完毕, (非local的一些变量, 则会在运行环境env中记录下来, 这个变量相对local来说是开放的,开放到环境中, local变量则生存在匿名函数中, 函数运行结束, 则local消失, 但是env中记录的非local变量得以保存)。

 

module中如何设置key为变量值的元素

在module文件中, 由于其运行环境是隐式的, 代码中无法直接同名字面量来引用到这个环境, 那么对于一种特殊情况, 需要在环境中记录一些key-value对, 但是key是来自于变量, 如何实现?

 

由于整个文件是一个chunk, 即匿名函数, 故可以借助getfenv,来获取运行环境。

local strkey = "vartwo"


-- 本模块环境
local env = getfenv(1)

env[strkey] = 2

 

module中如何引用调用者的环境

既然module文件是一个chunk对应一个一名函数, 则可以通过getfenv(2), 来获取此函数的调用者,

即谁require此文件的函数, 这个函数运行的环境, 即为调用者环境。

-- 调用此模块的环境
local env_caller = getfenv(2)

env["env_caller"] = env_caller

 

实验代码

module_test.lua

 

module(..., package.seeall)


varone = 1

local strkey = "vartwo"


-- 本模块环境
local env = getfenv(1)

env[strkey] = 2

print("vartwo="..vartwo)

 

-- 调用此模块的环境
local env_caller = getfenv(2)

env["env_caller"] = env_caller

 

 

main.lua

 


local testmodule = require("module_test")

var_caller = 55

print("testmodule.vartwo=" .. testmodule.vartwo)


local function printTable(tbl)

    for k,v in pairs(tbl) do
        print("kay="..tostring(k) .. "  value="..tostring(v))
    end

end


print("----------- testmodule ------------------")

printTable(testmodule)

 

print("-----------  testmodule.env_caller ------------------")

printTable(testmodule.env_caller)

 

 

运行输出结果:

>lua -e "io.stdout:setvbuf ‘no‘" "main.lua"
vartwo=2
testmodule.vartwo=2
----------- testmodule ------------------
kay=_NAME  value=module_test
kay=_PACKAGE  value=
kay=varone  value=1
kay=_M  value=table: 005F9410
kay=vartwo  value=2
kay=env_caller  value=table: 005F2650
-----------  testmodule.env_caller ------------------
kay=string  value=table: 005F9398
kay=xpcall  value=function: 005F6CE8
kay=module_test  value=table: 005F9410
kay=package  value=table: 005F7160
kay=tostring  value=function: 005F6CC8
kay=print  value=function: 005F6808
kay=os  value=table: 005F95A0
kay=unpack  value=function: 005F6AE8
kay=require  value=function: 005F8078
kay=getfenv  value=function: 005F6788
kay=setmetatable  value=function: 005F6C48
kay=next  value=function: 005F66E8
kay=assert  value=function: 005F68A8
kay=tonumber  value=function: 005F6CA8
kay=io  value=table: 005F94D8
kay=rawequal  value=function: 005F6828
kay=collectgarbage  value=function: 005F6668
kay=arg  value=table: 005F92D0
kay=getmetatable  value=function: 005F6688
kay=module  value=function: 005F7FD8
kay=rawset  value=function: 005F6A28
kay=var_caller  value=55
kay=math  value=table: 005F9348
kay=debug  value=table: 005F90F0
kay=pcall  value=function: 005F6988
kay=table  value=table: 005F72C8
kay=newproxy  value=function: 005F0568
kay=type  value=function: 005F6D08
kay=coroutine  value=table: 005F70C0
kay=_G  value=table: 005F2650
kay=select  value=function: 005F6708
kay=gcinfo  value=function: 005F69E8
kay=pairs  value=function: 005F0538
kay=rawget  value=function: 005F6908
kay=loadstring  value=function: 005F67C8
kay=ipairs  value=function: 005F2BD0
kay=_VERSION  value=Lua 5.1
kay=dofile  value=function: 005F67A8
kay=setfenv  value=function: 005F6A08
kay=load  value=function: 005F6888
kay=error  value=function: 005F6768
kay=loadfile  value=function: 005F6848
>Exit code: 0

lua module环境探秘

标签:

原文地址:http://www.cnblogs.com/lightsong/p/5743135.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!