Lua迭代器如何使用?

在Lua中,迭代器是一种用于遍历集合元素的重要工具。掌握迭代器的使用方法,对于提高Lua编程的效率和代码的可读性具有重要意义。

1.迭代器概述

12.1.1 迭代器介绍

迭代器是一种设计模式,它提供了一种访问集合元素的方法,而不需要暴露其底层结构。在Lua中,迭代器通常以一个函数的形式表示,每次调用该函数时,它会返回集合中的下一个元素。当没有更多元素时,迭代器返回nil。

1.2 泛型for循环

在Lua中,泛型for循环用于遍历一个迭代器函数返回的所有值。泛型for循环的基本语法如下: 

for variable1, variable2, ... in iterator_expression do-- 执行代码块
end

这里的iterator_expression通常是一个表达式,它应该返回三个值:迭代器函数、状态变量和初始值。在每一次循环中,Lua会调用迭代器函数,传递状态变量和上一次调用的结果(除了第一次,它会传递初始值)。当迭代器函数返回nil作为第一个值时,循环结束。

1.3 迭代器应用场景

迭代器在Lua中有多种应用场景,例如:

  • 遍历数组或表。
  • 遍历文件中的每一行。
  • 处理数据库查询结果等。

通过使用迭代器,可以更加简洁地处理这些任务,同时保持代码的可读性和可维护性。

2.内置迭代器

Lua提供了两个常用的内置迭代器:

迭代器

作用

pairs(t)

用于遍历表中的所有键值对,它的遍历顺序是不确定的(通常是按照哈希表内部的存储顺序)。

ipairs(t)

用于遍历数组,即从索引1开始直到遇到第一个nil元素为止。它返回两个值:索引和索引对应的元素。

这两个迭代器都是无状态的,意味着它们不保留任何额外的状态信息,只依赖于传递给它们的状态常量和控制变量。

假设有一个包含数字的数组,可以使用ipairs函数来遍历该数组。

-- ipairs_test.lua文件
local numbers = {10, 20, 30}for index, value in ipairs(numbers) doprint(index, value)
end

执行以上脚本代码,程序输出结果如下。

1   10
2   20
3   30

根据上面的示例,下面我们来看看泛型for的执行过程:

  • 首先,初始化,计算in后面表达式的值,表达式应该返回泛型for需要的三个值:迭代函数、状态常量、控制变量;与多值赋值一样,如果表达式返回的结果个数不足三个会自动用nil补足,多出部分会被忽略;
  • 第二,将状态常量和控制变量作为参数调用迭代函数(注意:对于for结构来说,状态常量没有用处,仅仅在初始化时获取它的值并传递给迭代函数);
  • 第三,将迭代函数返回的值赋给变量列表;
  • 第四,如果返回的第一个值为nil循环结束,否则执行循环体;
  • 第五,回到第二步再次调用迭代函数。

对于一个关联数组(哈希表,key-value键值形式),我们可以用pairs来遍历键值对。

-- pairs_test.lua文件
local person = {name = "Alice",age = 30,city = "Beijing"
}for key, value in pairs(person) doprint(key, value)
end

执行以上脚本代码,程序输出结果如下。

name  Alice
age   30
city  Beijing

注意,pairs遍历表元素的顺序不是固定的,如果你想按照特定顺序访问元素,你应该显式地排序或者使用有序的数据结构。

3.自定义迭代器

在Lua编程语言中,迭代器是一种允许程序员遍历数据集合的重要工具。它让开发者可以访问集合中的每一个元素,而无需暴露集合的内部结构。根据是否保存遍历过程的状态,迭代器被分为两种类型:无状态迭代器和多状态迭代器。下面我们将详细介绍这两种迭代器,并提供具体的语法和示例帮助理解。

3.1 无状态迭代器

无状态迭代器是指不保留任何状态信息的迭代器。这意味着,在每次调用迭代函数时,它并不依赖于之前的状态来决定下一个元素的位置。相反,它仅使用两个参数——状态常量(通常是表本身)和控制变量(如索引或键)——来计算并返回下一个元素。例如,ipairs函数就是一个典型的无状态迭代器,它遍历数组的每一个元素,且元素的索引必须是数值。

在Lua中,使用无状态迭代器时,for循环的表达式部分需要提供三个值:

  • 迭代函数:用于生成下一个元素。
  • 状态常量:在迭代过程中不会改变的值,通常是需要遍历的集合。
  • 控制变量:用于控制迭代过程的变量,通常在每次迭代后递增。

无状态迭代器的语法总结一下就是,当使用无状态迭代器时,泛型for循环会保存三个值:迭代函数、状态常量以及控制变量。这些值是在循环开始前由表达式计算出来的。每次迭代时,迭代函数都会以状态常量和控制变量作为参数被调用,并返回新的控制变量和一个或多个元素值。如果返回的第一个非空值为nil,则循环结束。

如下实现了一个简单的数字平方迭代器的例子。

-- no_state_iterator_test.lua文件
function square(iteratorMaxCount, currentNumber)if currentNumber < iteratorMaxCount thencurrentNumber = currentNumber + 1return currentNumber, currentNumber * currentNumberend
endfor i, n in square, 3, 0 doprint(i, n) -- 输出结果为: 1 1, 2 4, 3 9
end

在这个例子中,square函数充当了无状态迭代器的角色。它接收最大计数iteratorMaxCount和当前数字currentNumber作为参数,然后根据这两个值计算出下一个要返回的数字及其平方值。

执行以上脚本代码,程序输出结果如下。

1   1
2   4
3   9

另一个常见的无状态迭代器是ipairs,它是用来遍历数组形式的表的。其工作原理如下所示。

function ipairs(t)return iter, t, 0
endfunction iter(a, i)i = i + 1local v = a[i]if v thenreturn i, vend
end

这段代码展示了如何创建一个类似于ipairs的迭代器,其中iter函数是一个无状态迭代器,因为它只依赖传入的状态常量a和控制变量i来确定下一个元素的位置。

3.2 多状态迭代器

多状态迭代器会在迭代过程中保存额外的状态信息。这使得每个迭代器实例都可以独立地遍历相同的集合,而不会相互影响。为了实现这一点,通常会使用闭包或者将所有必要的状态封装在一个表内,作为迭代器的一部分。

  • 使用闭包:在迭代函数内部定义局部变量来保存状态,这些变量在每次迭代中都会被保留。
  • 使用表:将状态信息封装到一个表中,将表作为迭代器的状态常量传递给迭代函数。

使用闭包。下面是一个创建自定义迭代器的例子,该迭代器可以遍历一个表中的所有元素,同时计算并输出它们的长度。

-- multi_state_iterator_test.lua文件
-- 定义一个数组
local array = {"Google", "Baidu", "ByteDance"}-- 定义一个多状态迭代器
local function elementIterator(collection)local index = 0local count = #collection-- 返回一个闭包作为迭代器return function()index = index + 1if index <= count thenreturn collection[index], string.len(tostring(collection[index]))endend
end-- 使用迭代器遍历数组
for element, length in elementIterator(array) doprint(element, length)
end

在这个例子中,elementIterator函数创建了一个闭包,该闭包维护了一个私有的index计数器和count变量。每当迭代器被调用时,它就会递增index并检查是否超过了count的限制。如果没有超过,则返回当前元素及其字符串表示形式的长度。

执行以上脚本代码,程序输出结果如下。

Google    6
Baidu   5
ByteDance   9

此外,还可以使用表来保存状态信息。例如,如果我们想要遍历一个包含文件路径的表,并读取每个文件的内容,我们可以这样做。

local files = {"file1.txt", "file2.txt"}function fileReader(paths)local i = 0return function()i = i + 1local path = paths[i]if path thenlocal f = io.open(path)if f thenlocal content = f:read("*all")f:close()return path, contentendendend
endfor path, content in fileReader(files) doprint("File:", path)print("Content:", content)
end

这里,fileReader函数创建了一个闭包,该闭包负责打开指定路径下的文件并读取其内容。由于闭包能够捕获并记住外部作用域中的变量,因此它可以轻松地跟踪正在处理的是哪个文件。

最后对无状态迭代器和多状态迭代器做如下总结:

  • 无状态迭代器:不保留任何状态,每次迭代都只依赖于迭代函数、状态常量和控制变量。适用于简单的迭代场景,如遍历数组。
  • 多状态迭代器:需要保存多个状态信息,可以通过闭包或表来实现。适用于复杂的迭代场景,如遍历文件中的单词。

综上所述,无论是无状态还是多状态迭代器,它们都提供了灵活且强大的机制来遍历各种类型的集合。选择哪种迭代器取决于具体的应用场景和个人偏好。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/500666.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《计算机组成及汇编语言原理》阅读笔记:p177-p177

《计算机组成及汇编语言原理》学习第 13 天&#xff0c;p177-p177 总结&#xff0c;总计 1 页。 一、技术总结 1.real mode A programming model where the program has access to the entire capability of the machine, bypassing security and memory management. Useful…

珞珈一号夜光遥感数据地理配准,栅格数据地理配准

目录 一、夜光数据下载&#xff1a; 二、夜光遥感数据地理配准 三、计算夜光数据值 四、辐射定标 五、以表格显示分区统计 五、结果验证 夜光数据位置和路网位置不匹配&#xff0c;虽然都是WGS84坐标系&#xff0c;不匹配&#xff01;&#xff01;&#xff01;不要看到就直接…

python学习笔记—12—

1. 布尔类型 (1) 定义 (2) 比较运算符 (3) 代码演示 1. 手动定义 bool_1 True bool_2 False print(f"bool_1的内容是&#xff1a;{bool_1}, 类型是&#xff1a;{type(bool_1)}") print(f"bool_2的内容是&#xff1a;{bool_2}, 类型是&#xff1a;{type(bool…

Redis 使用redisTemplate获取某个规则下的key的全量数据(示例Set结构)

如下是redis中存储的数据结构 我想取key以favorites:结尾的所有数据 Redis 的 SCAN 命令用于迭代数据库中的键&#xff0c;支持通过模式过滤结果。模式规则基于 Redis 的通配符匹配语法&#xff0c;类似于文件名匹配规则&#xff1a; *&#xff1a;匹配零个或多个字符。?&…

CertiK《Hack3d:2024年度安全报告》(附报告全文链接)

CertiK《Hack3d&#xff1a;2024年度安全报告》现已发布&#xff0c;本次报告深入分析了2024年Web3.0领域的安全状况。2024年损失总额超过23亿美元&#xff0c;同比增幅高达31.61%&#xff1b;其中&#xff0c;12月的损失金额最少。过去一年&#xff0c;网络钓鱼攻击和私钥泄露…

AI知识库与用户行为分析:优化用户体验的深度洞察

在当今数字化时代&#xff0c;用户体验&#xff08;UX&#xff09;已成为衡量产品成功与否的关键指标之一。AI知识库作为智能客服系统的重要组成部分&#xff0c;不仅为用户提供快速、准确的信息检索服务&#xff0c;还通过用户行为分析&#xff0c;为产品优化提供了深度洞察。…

Vue3 + ElementPlus动态合并数据相同的单元格(超级详细版)

最近的新项目有个需求需要合并单元列表。ElementPlus 的 Table 提供了合并行或列的方法&#xff0c;可以参考一下https://element-plus.org/zh-CN/component/table.html 但项目中&#xff0c;后台数据返回格式和指定合并是动态且没有规律的&#xff0c;Element 的示例过于简单&…

CSS进阶和SASS

目录 一、CSS进阶 1.1、CSS变量 1.2、CSS属性值的计算过程 1.3、做杯咖啡 1.4、下划线动画 1.5、CSS中的混合模式(Blending) 二、SASS 2.1、Sass的颜色函数 2.2、Sass的扩展(extend)和占位符(%)、混合(Mixin) 2.3、Sass的数学函数 2.4、Sass的模块化开发 2.5、Sass…

概率论与数理统计

概率论占比更多&#xff0c;三分之二左右 数理统计会少一些 事件之间的概率 ab互斥&#xff0c;不是ab独立 古典概型吃高中基础&#xff0c;考的不会很多 条件概率公式&#xff0c;要记 公式不要全记&#xff0c;很多有名称的公式是通过基础公式转换而来的 目的在于解决一…

javaEE-多线程进阶-JUC的常见类

juc:指的是java.util.concurrent包&#xff0c;该包中加载了一些有关的多线程有关的类。 目录 一、Callable接口 FutureTask类 参考代码&#xff1a; 二、ReentrantLock 可重入锁 ReentrantLock和synchronized的区别&#xff1a; 1.ReentantLock还有一个方法&#xff1a…

智慧工地系统:建筑施工智能化管理的全新模式

智慧工地概述 智慧工地是将互联网的理念和和物联网的技术引入建筑工地&#xff0c;依托物联网、互联网、大数据、5G技术&#xff0c;建立云端数据平台&#xff0c;形成大数据的业务体系&#xff0c;打通一线操作与远程监管的链条&#xff0c;实现劳务、安全、环境、材料等各个…

FastAPI 统一接口响应(Json)模板的使用示例

目录 FastAPI 统一接口响应&#xff08;Json&#xff09;模板的使用示例 实际案例 实现方法 1. 创建统一响应格式的中间件 2. 将中间件添加到 FastAPI 应用中 3. 解释代码 4. 扩展和定制 5. 测试 FastAPI 统一接口响应&#xff08;Json&#xff09;模板的使用示例 实际…

Linux平台下实现的小程序-进度条

目录 1.换行、回车概念 2.缓冲区 2.1缓冲区 2.2强制刷新 3.进度条程序 Makefile文件 ProgressBar.h ProgressBar.c Main.c 执行结果 1.换行、回车概念 /n&#xff1a;换行回车&#xff08;\r&#xff1a;回车&#xff09; 2.缓冲区 如下图在vim编辑器中的命令模式下…

C++ 基础思维导图(一)

目录 1、C基础 IO流 namespace 引用、const inline、函数参数 重载 2、类和对象 类举例 3、 内存管理 new/delete 对象内存分布 内存泄漏 4、继承 继承权限 继承中的构造与析构 菱形继承 1、C基础 IO流 #include <iostream> #include <iomanip> //…

Java - 日志体系_Apache Commons Logging(JCL)日志接口库_桥接Logback 及 源码分析

文章目录 PreApache CommonsApache Commons ProperLogging &#xff08;Apache Commons Logging &#xff09; JCL 集成logbackPOM依赖配置文件 logback.xml使用 源码分析jcl-over-slf4j 的工作原理1. LogFactory 的实现2. SLF4JLogFactory 和 Log 的实例化过程3. SLF4JLog 和 …

中小企业如何进行数字化转型?

​在这个日新月异的数字时代&#xff0c;企业数字化转型已成为不可逆转的潮流与战略选择。大数据、云计算、人工智能、物联网等前沿科技正重塑着各行各业的面貌。面对激烈的市场竞争和不断变化的客户需求&#xff0c;中小企业作为国民经济的重要组成部分&#xff0c;数字化转型…

闪存知识科普-基本储存单元结构

概述&#xff1a; 闪存&#xff0c;即FlashMemory。其基本储存单元&#xff08;Memory Cell&#xff09;如下图所示。看起来有点像N沟道&#xff08;N-Channel&#xff09;MOS管&#xff0c;但比MOS管多一个悬浮闸&#xff08;Floating Gate&#xff09;。悬浮闸内可以储存电荷…

[江科大STM32] 第五集快速建立STM32工程模板——笔记

保存&#xff0c;进去选芯片型号&#xff0c;我们是F10C8T6 一个MD&#xff0c;还有所有.c.h 这里所有文件 这里所有文件

Elasticsearch:基础概念

一、什么是Elasticsearch Elasticsearch是基于 Apache Lucene 构建的分布式搜索和分析引擎、可扩展数据存储和矢量数据库。它针对生产规模工作负载的速度和相关性进行了优化。使用 Elasticsearch 可以近乎实时地搜索、索引、存储和分析各种形状和大小的数据。Elasticsearch 是…

安卓播放器TVbox或影视仓软件如何链接到xiaoya小雅超集?很详细的教程

前言 这里咱们说的安卓播放器软件指的是这个&#xff1a; 还有这个&#xff1a; 这两个软件只是个壳&#xff0c;你需要做的仅仅是把需要的内容填写到对应的位置即可。 开始这个教程之前&#xff0c;你需要先部署好小雅&#xff0c;如果没有部署小雅&#xff0c;这个教程基本…