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函数创建了一个闭包,该闭包负责打开指定路径下的文件并读取其内容。由于闭包能够捕获并记住外部作用域中的变量,因此它可以轻松地跟踪正在处理的是哪个文件。
最后对无状态迭代器和多状态迭代器做如下总结:
- 无状态迭代器:不保留任何状态,每次迭代都只依赖于迭代函数、状态常量和控制变量。适用于简单的迭代场景,如遍历数组。
- 多状态迭代器:需要保存多个状态信息,可以通过闭包或表来实现。适用于复杂的迭代场景,如遍历文件中的单词。
综上所述,无论是无状态还是多状态迭代器,它们都提供了灵活且强大的机制来遍历各种类型的集合。选择哪种迭代器取决于具体的应用场景和个人偏好。
相关文章:
Lua迭代器如何使用?
在Lua中,迭代器是一种用于遍历集合元素的重要工具。掌握迭代器的使用方法,对于提高Lua编程的效率和代码的可读性具有重要意义。 1.迭代器概述 12.1.1 迭代器介绍 迭代器是一种设计模式,它提供了一种访问集合元素的方法,而不需要…...
qt中如何判断字符串是否为数字,整数,浮点数?
在 Qt 中,可以使用多种方法来判断字符串是否为数字、整数或浮点数。Qt 提供了一些方便的字符串和数值处理函数,可以帮助你实现这些判断。以下是几种常见的方法: 1. 使用 QRegularExpression Qt 提供了 QRegularExpression 类,可…...

Oracle sql developer and Toad for Oracle set start DBMS output
Oracle sql developer Toad for Oracle...

【踩坑】SparkSQL union/unionAll 函数的去重问题
【踩坑】SparkSQL union/unionAll 函数的去重问题 测试数据 case class Employee(first_name:String)val employeeDF1 spark.createDataset(Seq( Employee("Mary"), Employee("Mandy"),Employee("Kurt") )) val employeeDF2 spark.createDat…...

域上的多项式环,整除,相通,互质
例1.已知 (R,,x)为域,请选出正确的说法:(A)(R,,x)也是整区; ABCD (B)R中无零因子; C)R在x运算上满足第一、二、三指数律; (D)R只有平凡理想; (E)R只有平凡子环。 域的特征: 域中,非0元素的加法周期 思考、在模7整数环R,中,…...

计算机毕业设计PyHive+Hadoop深圳共享单车预测系统 共享单车数据分析可视化大屏 共享单车爬虫 共享单车数据仓库 机器学习 深度学习
温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 作者简介:Java领…...
Julia语言的学习路线
Julia语言的学习路线 引言 在现代编程世界中,编程语言如同工具,各自具有独特的特点与优势。Julia语言自2012年发布以来,以其优越的性能和优雅的语法逐渐吸引了越来越多的数据科学家、工程师和研究人员的关注。在本篇文章中,我们…...

对计网大题的一些指正(中间介绍一下CDM的原理和应用)
目录 前言: (1)五层原理体系结构每层功能: 下面是文档的答案: 我在之前的博客里面有介绍过五层原理体系结构, 按理来说,第五层应该是应用层才对,而会话层的功能应该被放到应用层…...

UGUI 优化DrawCall操作记录(基于Unity2021.3.18)
UGUI中相同材质相同Shader相同贴图的UI元素可以合并DrawCall。 1.使用图集 Unity性能优化---使用SpriteAtlas创建图集进行批次优化_unity2021.3.33 spriteatlas优化-CSDN博客 2.Canvas的子物体在场景树中的索引位置和不同图集不影响UI合批且UI网格没有重叠,如下图…...

前端实现大文件上传(文件分片、文件hash、并发上传、断点续传、进度监控和错误处理,含nodejs)
大文件分片上传是前端一种常见的技术,用于提高大文件上传的效率和可靠性。主要原理和步骤如下 文件分片 确定分片大小:确定合适的分片大小。通常分片大小在 1MB 到 5MB 之间使用 Blob.slice 方法:将文件分割成多个分片。每个分片可以使用 Bl…...
es单机安装脚本自动化
背景 所有部署工作都可以由机器本身完成,并不需要人的参与,人唯一需要做的是把变量提取出来,进行赋值喂给脚本,然后脚本自己执行即可。下边是es单机安装的过程和脚本,由人变到脚本执行,方便理解。 步骤 1、解压es软件tar包。 2、cd至解压以后得config目录下,vim修改…...

Java 数据库连接 - Sqlite
Java 数据库连接 - Sqlite PS: 1. 连接依赖库:[sqlite-jdbc-xxx.jar](https://mvnrepository.com/artifact/org.xerial/sqlite-jdbc)(根据连接的数据库版本选择) 2. 支持一次连接执行多次sql语句; 3. 仅本地连接;使用说明: publ…...

CentOS — 目录管理
文章目录 一、目录结构二、切换目录三、查看目录四、创建目录五、复制目录六、剪切目录七、删除目录 目录也是一种文件。 蓝色目录,绿色可执行文件,红色压缩文件,浅蓝色链接文件,灰色其它文件, 点开头的是隐藏文件&…...

【第二部分--Python之基础】04 函数
1 定义函数 自定义函数的语法格式如下: 以英文半角冒号结尾 由于定义函数时的参数不是实际数据,会在调用函数时传递给它们实际数据,所以我们称定义函数时的参数为形式参数,简称形参:称调用函数时传递的实际数据为实际参数&#x…...
我们公司只有3个人,一个前端,一个后端
在当今这个数字化时代,各行各业都离不开互联网技术的支撑,而在这股技术浪潮中,小而美的创业公司如同雨后春笋般涌现,它们凭借着灵活高效、创新不断的特点,在市场中占有一席之地。 今天,就让我带你走进这样一…...

基于LabVIEW的BeamGage自动化接口应用
设置 National Instruments LabVIEW可执行程序需要被配置为使用.NET 4框架。.NET允许自定义可执行程序的运行方式。可通过以下方式实现: 在LabVIEW安装目录中创建一个名为LabVIEW.exe.config的文本文件(例如:C:\Program Files\National Ins…...

【AI编辑器】Cursor与DeepSeek模型的集成:提升开发效率的新选择
目录 一、为什么选择DeepSeek模型 1.1 模型参数与训练 1.2 技术创新 1、FP8格式介绍 2、FP8混合精度训练的优势 3、FP8混合精度训练的技术要点 4、FP8混合精度训练的应用与挑战 1.3 性能表现 1.4 应用与部署 1.5 争议与前景 二、注册DeepSeek账号并获取API Key 三、…...

vue2实现excel文件预览
一、插件 通过xlsx插件解析excel数据,对解析后的html组件进行渲染展示。 npm install xlsx 二、完整代码 <template><!-- excel文件预览 --><divelement-loading-text"拼命加载中"element-loading-spinner"el-icon-loading"…...
STM32 和 ESP32
STM32 和 ESP32 是两种不同的微控制器系列,它们分别由不同的制造商生产,并且针对的应用场景和特性也有所不同。尽管如此,两者也有一些共通点,因为它们都是用于嵌入式系统开发的微控制器平台。以下是关于 STM32 和 ESP32 的联系与区…...

R语言中的时间序列分析·
1 数据集说明 AirPassengers 1949~1960年每月乘坐飞机的乘客数 JohnsonJohnson Johnson&Johnson每股季度收入 nhtemp 康涅狄格州纽黑文地区从1912年至1971年每年的平均气温 Nile 尼罗河的流量 sunspots 1749年~1983年月平均太阳黑子数 2 相关包 xts、forecast、tser…...

深入浅出Asp.Net Core MVC应用开发系列-AspNetCore中的日志记录
ASP.NET Core 是一个跨平台的开源框架,用于在 Windows、macOS 或 Linux 上生成基于云的新式 Web 应用。 ASP.NET Core 中的日志记录 .NET 通过 ILogger API 支持高性能结构化日志记录,以帮助监视应用程序行为和诊断问题。 可以通过配置不同的记录提供程…...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》
在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中࿰…...
测试markdown--肇兴
day1: 1、去程:7:04 --11:32高铁 高铁右转上售票大厅2楼,穿过候车厅下一楼,上大巴车 ¥10/人 **2、到达:**12点多到达寨子,买门票,美团/抖音:¥78人 3、中饭&a…...

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放
简介 前面两期文章我们介绍了I2S的读取和写入,一个是通过INMP441麦克风模块采集音频,一个是通过PCM5102A模块播放音频,那如果我们将两者结合起来,将麦克风采集到的音频通过PCM5102A播放,是不是就可以做一个扩音器了呢…...

视频字幕质量评估的大规模细粒度基准
大家读完觉得有帮助记得关注和点赞!!! 摘要 视频字幕在文本到视频生成任务中起着至关重要的作用,因为它们的质量直接影响所生成视频的语义连贯性和视觉保真度。尽管大型视觉-语言模型(VLMs)在字幕生成方面…...

现代密码学 | 椭圆曲线密码学—附py代码
Elliptic Curve Cryptography 椭圆曲线密码学(ECC)是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础,例如椭圆曲线数字签…...

微服务商城-商品微服务
数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...

android RelativeLayout布局
<?xml version"1.0" encoding"utf-8"?> <RelativeLayout xmlns:android"http://schemas.android.com/apk/res/android"android:layout_width"match_parent"android:layout_height"match_parent"android:gravity&…...