彻底搞懂Lua闭包

一、基础:词法作用域

在深入闭包之前,我们必须先理解一个前置概念:词法作用域

Lua 具有词法作用域,这意味着变量的可见范围是由其声明所在的代码块决定的 。 在Lua 中,变量的作用域从声明开始,直到包含该声明的最内部块结束为止。 未用 local 关键字声明的变量默认为全局变量。

Lua 词法作用域的特点:

  • 局部变量: 使用 local 关键字声明的变量只在它们被声明的代码块(如 do...end 或函数体)内有效。

  • 全局变量: 不使用 local 声明的变量是全局变量,在整个程序中都可见。

  • 嵌套函数和闭包: 词法作用域使得内部函数可以访问外部函数的变量,这种机制是通过 upvalue 来实现的,它们共同构成了闭包。

  • 块作用域: Lua 没有传统意义上的块级作用域,只区分全局和局部作用域。 对于用 local 声明的变量,它们的作用域是词法确定的,而不是基于传统的块(如C++ 中的 iffor 块)。

二、闭包的实现:使用函数作为返回值

把内部函数作为 outer_function返回值

lua 复制代码
function make_printer()
    local outer_var = "I am remembered!"

    -- 创建并返回这个内部函数
    return function()
        print(outer_var)
    end
end

-- 1. 调用 make_printer,它返回了内部的那个匿名函数
local my_printer = make_printer()

-- 2. 注意!此时 make_printer() 已经执行完毕了!
--    理论上,它的局部变量 outer_var 应该已经被销毁了,但是因为内部的内部函数还在引用,所以并未真正销毁。

-- 3. 但当我们调用返回的函数时...
my_printer() -- 输出: I am remembered!

这就是一个最简单的闭包my_printer 不仅仅是一个函数,它是一个闭包实例 ,它"关闭并包围"了它需要的非局部变量 outer_var。这个被捕获的变量 outer_var 在 Lua 中被称为Upvalue

三、闭包是独立的:一座工厂,多个实例

闭包最强大的地方在于,每次调用外部的"工厂函数",都会创建一个全新的、独立的闭包,拥有自己独立的 Upvalue 实例。

让我们看一个经典的计数器例子:

lua 复制代码
function make_counter()
    local count = 0
    return function()
        count = count + 1
        return count
    end
end

-- 创建第一个计数器
local counter_A = make_counter()
print(counter_A()) -- 输出: 1
print(counter_A()) -- 输出: 2

-- 创建第二个、完全独立的计数器
local counter_B = make_counter()
print(counter_B()) -- 输出: 1 (它有自己的 count,从 0 开始)
print(counter_A()) -- 输出: 3 (counter_A 的 count 不受影响)
print(counter_B()) -- 输出: 2 

make_counter 就像一个工厂。每次调用它,都会生产出一个新的计数器。每个计数器都有自己私有的、互不干扰的 count 变量。count 对于外界是完全隐藏的,只能通过返回的那个函数来间接操作。

结语

点个赞,关注我获取更多实用 Lua 技术干货!如果觉得有用,记得收藏本文!

相关推荐
天***88963 小时前
Chrome扩展安装插件教程,Edge安装插件扩展教程,浏览器安装扩展程序方法
前端·chrome·edge
心.c4 小时前
深拷贝浅拷贝
开发语言·前端·javascript·ecmascript
IT_陈寒4 小时前
Vue 3.4性能优化实战:5个鲜为人知的Composition API技巧让打包体积减少40%
前端·人工智能·后端
前端九哥4 小时前
💻【急招!27届前端实习生】广州4399实习太幸福了!江景+三餐+健身房全都有😭
前端·面试·招聘
咖啡の猫5 小时前
Vue全局事件总线
前端·javascript·vue.js
Lovereo5 小时前
我的目标检测性能优化之路:预算不够、GPU 没有、但性能我得要
前端
蒙娜丽宁5 小时前
Rust 与 WebAssembly:构建高效前端应用的全流程复盘
前端·rust·wasm
这儿有一堆花5 小时前
使用 Actix-web 开发高性能 Web 服务
前端·数据库
豆苗学前端5 小时前
10分钟带你入门websocket,并实现一个在线多人聊天室
前端·javascript·后端