首页 > 其他分享 >Lua 模块与包

Lua 模块与包

时间:2023-06-19 21:00:50浏览次数:40  
标签:lua module --- Lua 模块 local

Lua 模块与包

模块类似于一个封装库,

从 Lua 5.1 开始,Lua 加入了标准的模块管理机制,

可以把一些公用的代码放在一个文件里,以 API 接口的形式在其他地方调用,

有利于代码的重用和降低代码耦合度。

Lua 的模块是由变量、函数等已知元素组成的 table

因此创建一个模块很简单,就是创建一个 table,

然后把需要导出的常量、函数放入其中,最后返回这个 table 就行。

以下为创建自定义模块 module.lua,文件代码格式如下:

---
--- Generated by EmmyLua(https://github.com/EmmyLua)
--- Created by 13355.
--- DateTime: 2023/6/19 20:07
---
-- 文件名为 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

由上可知,模块的结构就是一个 table 的结构,因此可以像操作调用 table 里的元素那样来操作调用模块里的常量或函数。

上面的 func2 声明为程序块的局部变量,

即表示一个私有函数,

因此是不能从外部访问模块里的这个私有函数,必须通过模块里的公有函数来调用.


require 函数

Lua提供了一个名为require的函数用来加载模块。

要加载一个模块,只需要简单地调用就可以了。例如:

require("<模块名>")

或者

require "<模块名>"

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

test_module.lua 文件

---
--- Generated by EmmyLua(https://github.com/EmmyLua)
--- Created by 13355.
--- DateTime: 2023/5/14 18:57
---
require("module")
print(module.constant)
module.func3()

以上代码执行结果为:

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

或者给加载的模块定义一个别名变量,方便调用:

test_module2.lua 文件

---
--- Generated by EmmyLua(https://github.com/EmmyLua)
--- Created by 13355.
--- DateTime: 2023/5/14 18:57
---
-- test_module.lua 文件
-- module 模块为上文提到到 module.lua
require("module")

print(module.constant)

module.func3()

以上代码执行结果为:

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

加载机制

对于自定义的模块,模块文件不是放在哪个文件目录都行,

函数 require 有它自己的文件路径加载策略,它会尝试从 Lua 文件或 C 程序库中加载模块。

require 用于搜索 Lua 文件的路径是存放在全局变量 package.path 中,

当 Lua 启动后,会以环境变量 LUA_PATH 的值来初始这个环境变量。

如果没有找到该环境变量,则使用一个编译时定义的默认路径来初始化。

当然,如果没有 LUA_PATH 这个环境变量,也可以自定义设置,

在当前用户根目录下打开 .profile 文件(没有则创建,打开 .bashrc 文件也可以),

例如把 "~/lua/" 路径加入 LUA_PATH 环境变量里:

#LUA_PATH
export LUA_PATH="~/lua/?.lua;;"

文件路径以 ";" 号分隔,最后的 2 个 ";;" 表示新加的路径后面加上原来的默认路径。

接着,更新环境变量参数,使之立即生效。

source ~/.profile

这时假设 package.path 的值是:

/Users/dengjoe/lua/?.lua;./?.lua;/usr/local/share/lua/5.1/?.lua;/usr/local/share/lua/5.1/?/init.lua;/usr/local/lib/lua/5.1/?.lua;/usr/local/lib/lua/5.1/?/init.lua

那么调用 require("module") 时就会尝试打开以下文件目录去搜索目标。

/Users/dengjoe/lua/module.lua;
./module.lua
/usr/local/share/lua/5.1/module.lua
/usr/local/share/lua/5.1/module/init.lua
/usr/local/lib/lua/5.1/module.lua
/usr/local/lib/lua/5.1/module/init.lua

如果找过目标文件,则会调用 package.loadfile 来加载模块。否则,就会去找 C 程序库。

搜索的文件路径是从全局变量 package.cpath 获取,而这个变量则是通过环境变量 LUA_CPATH 来初始。

搜索的策略跟上面的一样,只不过现在换成搜索的是 so 或 dll 类型的文件。

如果找得到,那么 require 就会通过 package.loadlib 来加载它。


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 库了。

标签:lua,module,---,Lua,模块,local
From: https://www.cnblogs.com/hcgk/p/17492167.html

相关文章

  • 03后台主页模块设计,simpleui后台管理,轮播图接口,跨域问题详解,前台主页功能
    1后台主页模块设计#1创建后台主页模块(一个模块一个app)python../../manage.pystartapphome#2在models中写轮播图表 -写一个基表BaseModel-写轮播图表#3迁移###BaseModel##########fromdjango.dbimportmodelsclassBaseModel(models.Model):creat......
  • 旅游网管理系统(系统模块设计)-自我训练
    旅游网是一款常见的旅游网站,项目分为管理员端和用户端。管理员端(后台)主要进行旅游产品的维护,用户端(前台)主要进行旅游产品的展示。管理员端的主要功能有:管理员管理角色管理权限管理认证和授权产品类型管理旅游产品管理用户端的主要功能有:用户注册用户登录查询旅游产......
  • Ansible模块介绍
    转自:https://blog.csdn.net/weixin_53388991/article/details/127533067 ......
  • fcntl文件枷锁模块
    fcntl模块本模块基于文件描述符来进行文件控制和I/O控制。它是Unix系统调用fcntl()和ioctl()的接口。关于这些调用的完整描述,请参阅Unix手册的fcntl(2)和ioctl(2)页面。flock介绍fcntl.flock(f,operation)f:文件描述符operation:操作fcntl.LOCK_UN......
  • Lua语言- 1
    0-Lua语言介绍特性轻量级:它用标准C语言编写并以源代码形式开放,编译后仅仅一百余K,可以很方便的嵌入别的程序里。可扩展:Lua提供了非常易于使用的扩展接口和机制:由宿主语言(通常是C或C++)提供这些功能,Lua可以使用它们,就像是本来就内置的功能一样。其它特性:支持面向过程(proce......
  • nodejs 伪全局变量模块
    使用这个文件可以实现不同文件中读写变量,适合当做共享变量文件名:globals.tsletglobals:any={myGlobal:{value:'canbeanytype:String,Array,Object,...'},aReadonlyGlobal:{value:'thisvalueisreadonly',protected:......
  • BOSHIDA AC DC电源模块在工业控制器的应用
    BOSHIDAACDC电源模块在工业控制器的应用AC/DC电源模块是一种广泛应用于工业控制器中的电源设备,它的作用是将交流电源转换为直流电源,为工业控制器提供稳定可靠的电源供应。在工业控制器中,AC/DC电源模块的应用不仅可以提高系统的稳定性和可靠性,还可以实现节能和环保,降低系统维护......
  • 关于Linux系统下Lua编程运行环境的部署安装
    这里以操作系统:RedHatEnterpriseLinuxrelease8.7(Ootpa)为例,讲解如下部署Lua编程脚本的运行环境首先对于Lua脚本,需要保证系统中有lua二进制程序文件,即/usr/bin/lua但最小化安装的Linux- RedHatEnterpriseLinuxrelease8.7(Ootpa) 笔者试了一下,没有lua命令了 ......
  • nrf52832学习-app_timer模块
    简单使用:app_timer是模块,使用时需要在sdk_config.h文件中将其使能添加头文件app_timer.h定义Timer定时器唯一识别号_my_timer_id和需要的定时时间常量MY_TIMER_INTERVALAPP_TIMER_DEF(_my_timer_id);#defineMY_TIMER_INTERVALAPP_TIMER_TICKS(_milliseconds)......
  • nrf52832学习-app_scheduler模块
    软件调度模块,但不是用来做任务调度的,而是为了在main函数中执行中断任务。说白了就是一个队列,中断中将数据和执行的函数放入队列中,然后main函数里一直去调用查询去执行,从而不阻塞中断app_sched_execute用来在main函数中执行,一直调用app_sched_event_put函数是在中断中调用将任务添......