当前位置: 首页 > web >正文

Lua中使用module时踩过的坑

       在lua中设置某个全局对象(假如对象名为LDataUser)为nil时, LDataUser并不会变成nil, 但在有些情况下设置LDataUser = nil时却真变成了nil,然后会导致后续再使用LDataUser时会抛nil异常, 后来发现是使用module搞的鬼,下面看看豆包AI给的解释,还是很详细的,AI越来越厉害啦.

问:

LDataUser = nil这行代码在module和package.seeall定义的lua文件中并不会变为nil, 但是在非module和package.seeall定义的lua文件中会变成nil, 是为什么?

豆包:

这种现象与Lua的模块作用域和环境变量( _ENV )的管理机制有关,具体原因如下:

### 1. 核心差异:变量存储的作用域不同
- 使用 module 或 package.seeall 的文件 : 这类文件通常通过 module 函数或 package.seeall 显式设置了模块的环境表( _ENV )。此时模块内的「全局变量」(如 LDataUser )实际存储在模块自身的环境表中(而非全局环境 _G )。因此执行 LDataUser = nil 时,只是将模块环境表中的该变量置为 nil ,但全局环境 _G 中的 LDataUser 可能仍保留原值(如果其他文件在全局环境中定义过它)。
- 非 module / package.seeall 的文件 : 这类文件的「全局变量」默认存储在全局环境 _G 中。执行 LDataUser = nil 会直接修改 _G 中的该变量,因此其他文件通过全局环境访问 LDataUser 时会看到 nil 。
### 2. 示例说明
假设 A.lua 使用 module 定义, B.lua 不使用:

```
-- B.lua(不使用 module)
LDataUser = { name = "user2" }  -- 存储在全局环境 _G 中

function B.clearGlobalUser()
    LDataUser = nil  -- 清空 _G 中的 LDataUser
end
```
此时:

- 调用 A.clearUser() 后, A 模块内的 LDataUser 为 nil ,但 _G.LDataUser 仍为 { name = "user2" } (其他文件通过全局访问时不受影响)。
- 调用 B.clearGlobalUser() 后, _G.LDataUser 被置为 nil ,所有文件通过全局访问 LDataUser 都会得到 nil 。
### 3. 关键总结
- module / package.seeall 通过修改模块的环境表( _ENV ),将模块内的「全局变量」隔离到模块自身的作用域中。
- 直接修改模块内的变量(如 LDataUser = nil )仅影响模块环境表,不影响全局环境 _G 。
- 非模块文件的「全局变量」直接存储在 _G 中,修改会影响所有依赖全局环境的代码。

问:

接着上面的疑问继续,假如使用module的A.lua中执行代码LDataUser = { name1 = "user1" } ,然后在使用了module的B.lua中执行代码LDataUser = { name2 = "user2" }, 最后在B.lua中执行了 LDataUser = nil,那么LDataUser中的name1也会为nil吗?就是说清理了所有module定义的lua文件在LDataUser中添加的属性,还是仅仅清理当前module定义的lua对象中新增的属性?

豆包:

在Lua中使用 module 定义的模块文件(如A.lua和B.lua)中, LDataUser 的赋值和置 nil 行为与模块的环境表( _ENV )密切相关,具体分析如下:

### 1. 模块环境隔离机制
每个通过 module 定义的Lua文件(如A.lua、B.lua)会创建独立的环境表( _ENV )。模块内的「全局变量」(如 LDataUser )实际存储在模块自身的环境表中,而非全局环境 _G 。因此:

- A.lua中的 LDataUser 存储在A模块的环境表(记为 _ENV_A )中;
- B.lua中的 LDataUser 存储在B模块的环境表(记为 _ENV_B )中。
### 2. 赋值与置nil的作用范围
- A.lua执行 LDataUser = { name1 = "user1" } :仅修改 _ENV_A 中的 LDataUser , _ENV_B 和 _G 中的 LDataUser 不受影响。
- B.lua执行 LDataUser = { name2 = "user2" } :仅修改 _ENV_B 中的 LDataUser , _ENV_A 和 _G 中的 LDataUser 不受影响。
- B.lua执行 LDataUser = nil :仅将 _ENV_B 中的 LDataUser 置为 nil , _ENV_A 中的 LDataUser (即 name1 )仍保留原值。
### 3. 结论
在B.lua中执行 LDataUser = nil 不会清理A.lua中 LDataUser 的属性 (如 name1 )。每个模块的 LDataUser 独立存储在各自的环境表中,修改一个模块的 LDataUser 不会影响其他模块的同名变量。

http://www.xdnf.cn/news/6708.html

相关文章:

  • 计算机指令分类和具体的表示的方式
  • 【Win32 API】 lstrcmpA()
  • Java内存泄露生产环境排查过程,通透了
  • 计算机网络 : Socket编程
  • EXCEL在一列数据前统一添加负号
  • 6种方式来探究数据集的的方法worldquant
  • STM32外设AD-定时器触发 + DMA读取模板
  • RKNN开发环境搭建(ubuntu22.04)
  • 网络世界的“百变身份“:动态IP让连接更自由
  • 解锁DeepSeek潜能:Docker+Ollama打造本地大模型部署新范式
  • 【Python 操作 MySQL 数据库】
  • maven和npm区别是什么
  • 几种排序方式的C语言实现(冒泡、选择、插入、希尔等)
  • 大数据技术的主要方向及其应用详解
  • 【问题排查】easyexcel日志打印Empty row!
  • DeepSearch代表工作
  • 时钟产生的公共模块示例
  • Java 泛型与类型擦除:为什么解析对象时能保留泛型信息?
  • 随笔:hhhhh
  • Redisson 四大核心机制实现原理详解
  • 涂色不踩雷:如何优雅解决 LeetCode 栅栏涂色问题
  • Vue3项目使用ElDrawer后select方法不生效
  • 突围“百机大战”,云轴科技ZStack智塔获IDC中国AI大模型一体机推荐品牌
  • 第五章:Linux用户管理
  • 【无标题】威灏光电哲讯科技MES项目启动会圆满举行
  • leetcode 57. Insert Interval
  • Node.js 同步加载问题详解:原理、危害与优化策略
  • Spring Cloud动态配置刷新:@RefreshScope与@Component的协同机制解析
  • Gitee DevOps:中国企业数字化转型的加速引擎
  • UNiAPP地区选择