从Lua 5.1开始,我们可以使用require和module函数来获取和创建Lua中的模块。从使用者的角度来看,一个模块就是一个程序库,可以通过require来加载,之后便得到一个类型为table的全局变量。此时的table就像名字空间一样,可以访问其中的函数和常量,如:
复制代码 代码如下:
require "mod"
mod.foo()
local m2 = require "mod2"
local f = mod2.foo
f() 

1. require函数:

    require函数的调用形式为require "模块名"。该调用会返回一个由模块函数组成的table,并且还会定义一个包含该table的全局变量。在使用Lua中的标准库时可以不用显示的调用require,因为Lua已经预先加载了他们。

    require函数在搜素加载模块时,有一套自定义的模式,如:
 复制代码 代码如下:
    "sql",将会打开以下的文件:
 复制代码 代码如下:
    sql
    sql.lua
    c:/windows/sql
    /usr/local/lua/sql/sql.lua
 
    Lua将require搜索的模式字符串放在变量package.path中。当Lua启动后,便以环境变量LUA_PATH的值来初始化这个变量。如果没有找到该环境变量,则使用一个编译时定义的默认路径来初始化。如果require无法找到与模块名相符的Lua文件,就会找C程序库。C程序库的搜索模式存放在变量package.cpath中。而这个变量则是通过环境变量LUA_CPATH来初始化的。
   
2. 编写模块的基本方法:

    见如下代码和关键性注释:
复制代码 代码如下:
--将模块名设置为require的参数,这样今后重命名模块时,只需重命名文件名即可。
local modname = ...
local M = {}
_G[modname] = M

M.i = {r = 0, i = 1}  --定义一个模块内的常量。
function M.new(r,i) return {r = r, i = i} end
function M.add(c1,c2)
    return M.new(c1.r + c2.r,c1.i + c2.i)
end

function M.sub(c1,c2)
    return M.new(c1.r - c2.r,c1.i - c2.i)
end
--返回和模块对应的table。
return M

3. 使用环境:

仔细阅读上例中的代码,我们可以发现一些细节上问题。比如模块内函数之间的调用仍然要保留模块名的限定符,如果是私有变量还需要加local关键字,同时不能加模块名限定符。如果需要将私有改为公有,或者反之,都需要一定的修改。那又该如何规避这些问题呢?我们可以通过Lua的函数“全局环境”来有效的解决这些问题。见如下修改的代码和关键性注释:
复制代码 代码如下:
--模块设置和初始化。这一点和上例一致。
local modname = ...
local M = {}
_G[modname] = M

--声明这个模块将会用到的全局函数,因为在setfenv之后将无法再访问他们,
--因此需要在设置之前先用本地变量获取。
local sqrt = mat.sqrt
local io = io

--在这句话之后就不再需要外部访问了。
setfenv(1,M)

--后面的函数和常量定义都无需模块限定符了。
i = {r = 0, i = 1}
function new(r,i) return {r = r, i = i} end
function add(c1,c2)
    return new(c1.r + c2.r,c1.i + c2.i)
end
 
function sub(c1,c2)
    return new(c1.r - c2.r,c1.i - c2.i)
end
--返回和模块对应的table。
return M

4. module函数:

    在Lua 5.1中,我们可以用module(...)函数来代替以下代码,如:
复制代码 代码如下:
local modname = ...
local M = {}
_G[modname] = M
package.loaded[modname] = M
    --[[
    和普通Lua程序块一样声明外部函数。
    --]]
setfenv(1,M)

由于在默认情况下,module不提供外部访问,必须在调用它之前,为需要访问的外部函数或模块声明适当的局部变量。然后Lua提供了一种更为方便的实现方式,即在调用module函数时,多传入一个package.seeall的参数,如:
复制代码 代码如下:
    module(...,package.seeall)


注意:5.2已经不支持module了,去看看lua官网的文档,没有这个函数了,我用package.loaded.module_name = _ENV来创建模块

标签:
Lua,教程,模块与包

免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
无争山庄资源网 Copyright www.whwtcm.com

评论“Lua教程(十一):模块与包详解”

暂无“Lua教程(十一):模块与包详解”评论...

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。