模块类似于一个封装库,从 Lua 5.1 开始,Lua 加入了标准的模块管理机制,可以把一些公用的代码放在一个文件里,以 API 接口的形式在其他地方调用,有利于代码的重用和降低代码耦合度。

Lua 的模块是由变量、函数等已知元素组成的 table,因此创建一个模块很简单,就是创建一个 table,然后把需要导出的常量、函数放入其中,最后返回这个 table 就行。以下为创建自定义模块 module.lua,文件代码格式如下:

-- 定义一个名为 module 的模块
module = {}
 
-- 定义一个常量
module.constant = "这是一个常量"
 
-- 定义一个函数
function module.func1()
    io.write("这是一个公有函数!\n")
end
 
local function func2()
    print("这是一个私有函数!")
end
 
function module.func3()
    func2()
end
 
return module

1.require 函数

Lua提供了一个名为require的函数用来加载模块。要加载一个模块,只需要简单地调用就可以了。例如:


require("<模块名>")


或者


require "<模块名>"


 执行 require 后会返回一个由模块常量或函数组成的 table,并且还会定义一个包含该 table 的全局变量。

require("module")
 
print(module.constant)
 
module.func3()

以上代码执行结果为:


这是一个常量 这是一个私有函数!


--1: 第一个代码是有lua虚拟机第一个解释执行的lua文件;
--2:第一个lua文件可以去包含和导入其他的lua文件;
    使用require关键字来导入其他lua
--3: require的时候就会装载lua文件,并执行lua代码;
    require“./test” // 不用文件的后缀名;
    require “文件夹/lua文件名”;
    require “文件夹.lua文件名”;
--4: lua代码中的 return  语句:
    require “lua代码” 执行的时候如果有return 语句,那么require就返回return的对象;
--5: lua导出函数:
      将lua代码要到处的函数,做成一个表,把这个表返回出去;
--6: 如果在其他的地方使用又要装载一次;
--7: 如果require过了的lua代码,直接返回返回值,不再装载代码并执行;

2.module与package

--1: 默认情况下lua代码编写的函数是不能被外部识别的;

--2: 包: module(包名, package.seeall)

--3: 外部require 包名

    require “包名”

--4: 全局require模块后,其他的lua文件无需再去require了,方便使用;

3.C 包

Lua和C是很容易结合的,使用 C 为 Lua 写包。

与Lua中写包不同,C包在使用以前必须首先加载并连接,在大多数系统中最容易的实现方式是通过动态连接库机制。

Lua在一个叫loadlib的函数内提供了所有的动态连接的功能。这个函数有两个参数:库的绝对路径和初始化函数。所以典型的调用的例子如下:

local path = "/usr/local/lua/lib/libluasocket.so"
local f = loadlib(path, "luaopen_socket")

loadlib 函数加载指定的库并且连接到 Lua,然而它并不打开库(也就是说没有调用初始化函数),反之他返回初始化函数作为 Lua 的一个函数,这样我们就可以直接在Lua中调用他。

如果加载动态库或者查找初始化函数时出错,loadlib 将返回 nil 和错误信息。我们可以修改前面一段代码,使其检测错误然后调用初始化函数:

local path = "/usr/local/lua/lib/libluasocket.so"
-- 或者 path = "C:\\windows\\luasocket.dll",这是 Window 平台下
local f = assert(loadlib(path, "luaopen_socket"))

f()  -- 真正打开库

一般情况下我们期望二进制的发布库包含一个与前面代码段相似的 stub 文件,安装二进制库的时候可以随便放在某个目录,只需要修改 stub 文件对应二进制库的实际路径即可。

将 stub 文件所在的目录加入到 LUA_PATH,这样设定后就可以使用 require 函数加载 C 库了。