Lua基础之模块与require

2024-09-01 20:32
文章标签 基础 模块 lua require

本文主要是介绍Lua基础之模块与require,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

模块

Lua可以利用table实现模块加载

方法:

  1. 从require传入的参数中获取模块名或直接自定义变量名;
  2. 初始化一个空table;
  3. 在全局环境_G中添加模块名对应的字段,将空table赋值给这个字段;
  4. package.loaded中设置该模块;
  5. 设置环境变量。

例如:

local moduleName = ...     -- 模块名,可以在这里直接指定local M = {}               -- 初始化table
_G[moduleName] = M         -- 将这个局部变量最终赋值给模块名
package.loaded[moduleName] = M    -- 赋值给已经加载的tablesetmetatable(M, {__index = _G})   
--[[
元表会有一定的性能开销,也可以用local _G = _G保存全局的环境变量或者将需要用到的库先赋值
local math= math            -- 在我们自己的模块中需要用到math库,所以就先保存下来
local io = io               -- 需要用到io库,也保存下来
--]]setfenv(1, M)               -- 设置环境变量

 

利用module函数实现模块加载

Lua5.1提供了module函数,直接利用该函数就可以实现模块加载

module(name, cb1, cb2, ...)

module运行时:

     1. 如果package.loaded[name]是一个table,那么就把这个table作为一个module返回 

     2. 如果全局变量name是一个table,就把这个全局变量作为一个module返回

     3. 创建一个新的table: 

t = 
{[name]=package.loaded[name],["_NAME"]=name,["_M"]=t,["_PACKAGE"]=*name*  -- 删除了最后的".XXXX"部分
}-- 例如
"hello.world"
t =
{["hello"]={["world"]={XXXXXXX}}
}

     4. 依次调用cbs:cb1(mod), cb2(mod),...将当前模块的环境设置为module,同时把package.loaded[name] = module

 

module 指令运行完后,整个环境被压栈,所以前面全局的东西无法访问,这种情况可以这样解决的:

    a. 使用module(..., package.seeall)加载模块,这句话的功能就好比之前的功能再加上了setmetatable(M, {__index = _G})
    b. 在module前local _G=_G
    c. 将需要用到的库先赋值 如:local print= print

 

require

require的加载路径

?;?.lua;c:\windows\?;/usr/local/lua/?/?.lua

搜索一个文件时,在windows上,很多都是根据windows的环境变量path来搜索,而require所使用的路径与传统的路径不同,require采用的路径是一连串的模式,其中每项都是一种将模块名转换为文件名的方式。require会用模块名来替换每个“?”,然后根据替换的结果来检查是否存在这样一个文件,如果不存在,就会尝试下一项。
假如尝试搜索Test.lua时,require就会依次搜索下面的路径,直至搜索到该lua文件,

Test
Test.lua
c:\windows\Test
/usr/local/lua/mod/Test.lua

 

require加载过程

用法:

require(name)

加载顺序:

       1. 首先在package.loaded查找name,如果该模块已经存在,就直接返回它的值;

       2. 在package.preload查找name, 如果preload存在,那么就把它作为loader,调用loader(L);

       3. 根据package.path的模式查找lua库name,这个库是通过module函数定义的,对于顶层的lua库,文件名和库名是一样的而且不需要调用显式地在lua文件中调用module函数,也就是说lua会根据lua文件直接完成一个loader的初始化过程;

       4. 根据package.cpath查找c库,这个库是符合lua的一些规范的(export具有一定特征的函数接口),lua先已动态的方式加载该c库,然后在库中查找并调用相应名字的接口,例如:luaopen_hello_world;

       5. 以第一个"."为分割,将模块名划分为:(main, sub)的形式,根据package.cpath查找main,如果存在,就加载该库并查询相应的接口:luaopen_main_sub,例如:先查找hello库,并查询luaopen_hello_world接口

       6. 得到loader后,用name作为唯一的参数调用该loader函数。当然参数是通过lua的栈传递的,所以loader的原型必须符合lua的规范:int LUA_FUNC(lua_State *L)。

       如果找到的是一个C程序库,就通过loadlib来加载。loadfile和loadlib都只是加载了代码,并没有运行它们,为了运行代码,require会以模块名作为参数来调用这些代码

       require会将这个loader的返回值赋给package.loaded[modelname],如果loader不返回值同时package.loaded[modelname]不存在时,require就会把package.loaded[modelname]设为true。最后reuqire把package.loaded[modelname]返回给调用者。

 

注:

package.path:保存加载外部模块的搜索路径,用于搜索自己写的库文件或者第三方的库文件package.cpath:作用和packag.path一样,但它是用于加载第三方c库的。初始值可以通过环境变量LUA_CPATH来设置,用于搜索自己写的so库文件或者第三方的so库文件

 

参考链接:

https://blog.codingnow.com/2006/02/lua_51_module.html

https://www.jb51.net/article/55818.htm

https://www.cnblogs.com/yyxt/p/3870236.html

https://blog.csdn.net/xiejunna/article/details/72874981

 

这篇关于Lua基础之模块与require的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1127968

相关文章

python: 多模块(.py)中全局变量的导入

文章目录 global关键字可变类型和不可变类型数据的内存地址单模块(单个py文件)的全局变量示例总结 多模块(多个py文件)的全局变量from x import x导入全局变量示例 import x导入全局变量示例 总结 global关键字 global 的作用范围是模块(.py)级别: 当你在一个模块(文件)中使用 global 声明变量时,这个变量只在该模块的全局命名空

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

AI基础 L9 Local Search II 局部搜索

Local Beam search 对于当前的所有k个状态,生成它们的所有可能后继状态。 检查生成的后继状态中是否有任何状态是解决方案。 如果所有后继状态都不是解决方案,则从所有后继状态中选择k个最佳状态。 当达到预设的迭代次数或满足某个终止条件时,算法停止。 — Choose k successors randomly, biased towards good ones — Close

音视频入门基础:WAV专题(10)——FFmpeg源码中计算WAV音频文件每个packet的pts、dts的实现

一、引言 从文章《音视频入门基础:WAV专题(6)——通过FFprobe显示WAV音频文件每个数据包的信息》中我们可以知道,通过FFprobe命令可以打印WAV音频文件每个packet(也称为数据包或多媒体包)的信息,这些信息包含该packet的pts、dts: 打印出来的“pts”实际是AVPacket结构体中的成员变量pts,是以AVStream->time_base为单位的显

C 语言基础之数组

文章目录 什么是数组数组变量的声明多维数组 什么是数组 数组,顾名思义,就是一组数。 假如班上有 30 个同学,让你编程统计每个人的分数,求最高分、最低分、平均分等。如果不知道数组,你只能这样写代码: int ZhangSan_score = 95;int LiSi_score = 90;......int LiuDong_score = 100;int Zhou

Jenkins构建Maven聚合工程,指定构建子模块

一、设置单独编译构建子模块 配置: 1、Root POM指向父pom.xml 2、Goals and options指定构建模块的参数: mvn -pl project1/project1-son -am clean package 单独构建project1-son项目以及它所依赖的其它项目。 说明: mvn clean package -pl 父级模块名/子模块名 -am参数

寻迹模块TCRT5000的应用原理和功能实现(基于STM32)

目录 概述 1 认识TCRT5000 1.1 模块介绍 1.2 电气特性 2 系统应用 2.1 系统架构 2.2 STM32Cube创建工程 3 功能实现 3.1 代码实现 3.2 源代码文件 4 功能测试 4.1 检测黑线状态 4.2 未检测黑线状态 概述 本文主要介绍TCRT5000模块的使用原理,包括该模块的硬件实现方式,电路实现原理,还使用STM32类

c++基础版

c++基础版 Windows环境搭建第一个C++程序c++程序运行原理注释常亮字面常亮符号常亮 变量数据类型整型实型常量类型确定char类型字符串布尔类型 控制台输入随机数产生枚举定义数组数组便利 指针基础野指针空指针指针运算动态内存分配 结构体结构体默认值结构体数组结构体指针结构体指针数组函数无返回值函数和void类型地址传递函数传递数组 引用函数引用传参返回指针的正确写法函数返回数组