文章目录
- 一、前言
- 二、Lua 数组
- (2.1)一维数组
- (2.1.1)初始化:
- (2.1.2)计算数组长度:
- (2.1.3)访问数组元素:
- (2.1.4)数组添加元素
- (2.1.5)数组删除元素
- (2.1.6)数组修改元素
- (2.2)多维数组
- (2.2.1)阵列多维数组
- (2.2.2)其他多维数组
- 三、Lua 迭代器
- (3.1)泛型 for 迭代器
- (3.1.1) 无状态迭代器
- (3.1.2)多状态迭代器
一、前言
今天继续系统学习Lua。这个系列是我系统学习Lua语言的学习笔记,我会把遇到的一些比较值得记录与关注的知识写在里面,供自己以后进行回顾。
二、Lua 数组
数组是由相同的数据类型的元素按照一定顺序排列的集合,可以是一维数组和多维数组。
在 Lua 中。数组不是一种特定的数据类型,而是一种用来存储同种数据类型的数据结构, Lua 使用table
的数据结构来实现数组的功能。
Lua 数组的索引键值可以使用整数来表示,数组的大小不是固定的, Lua 索引值是以1
开始的,但是也可以指定为0开始。
(2.1)一维数组
一维数组的逻辑结构是线性表。
(2.1.1)初始化:
(2.1.2)计算数组长度:
可以使用 #
计算数组的长度。
(2.1.3)访问数组元素:
可以使用 Lua 循环 遍历数组元素,也可以使用整数索引来访问数组元素,如果指定的索引没有值则返回 nil
。此外还可以是用负数作为数组的索引值。
(2.1.4)数组添加元素
可以使用索引向数组中添加元素。
(2.1.5)数组删除元素
可以使用 table.remove()
函数删除数组中的元素。
(2.1.6)数组修改元素
可以使用索引修改数组中的元素。
(2.2)多维数组
多维数组即数组中包含数组或一维数组的索引键对应一个数组。
(2.2.1)阵列多维数组
以下是一个三行三列的阵列多维数组:
(2.2.2)其他多维数组
不同索引键的三行三列阵列多维数组:
三、Lua 迭代器
迭代器(iterator)是一种对象,它能够用来遍历标准模板库容器中的部分或全部元素,每个迭代器对象代表容器中的确定的地址。
在 Lua 中迭代器是一种支持指针类型的结构,它可以遍历集合的每一个元素。
(3.1)泛型 for 迭代器
泛型 for 在自己内部保存迭代函数,实际上它保存了三个值:迭代函数、状态常量、控制变量。
泛型 for 迭代器的语法如下:
上述代码中,k, v为变量列表;pairs(t) 为表达式列表。
泛型 for 的执行过程如下:
- 初始化,计算 in 后面表达式的值,表达式返回迭代函数、状态常量、控制变量这三个值;与多值赋值一样,如果表达式返回的值个数不足三个自动用 nil 不足,多于三个则多出的部分会被忽略。
- 将状态常量和控制变量作为参数调用迭代函数。(对于 for 循环来说,状态常量没有用处,仅仅在初始化时获取它的值并传递给迭代函数)。
- 将迭代函数返回的值赋给变量列表。
- 如果返回的第一个值为 nil 循环结束,否则执行循环体。
- 回到第二步再次调用迭代函数。
在 Lua 中常常使用函数来描述迭代器,每次调用该函数就返回集合的下一个元素。Lua 的迭代器包含以下两种类型:
1.无状态迭代器
2.多状态迭代器
(3.1.1) 无状态迭代器
无状态迭代器是指不保留任何状态的迭代器,因此在循环中可以利用无状态迭代器避免船舰闭包花费额外的代价。
每一次迭代,迭代函数都是用两个变量(状态常量和控制变量)的值作为参数调用,一个无状态的迭代器值利用这两个值可以获取下一个元素。
这种无状态迭代器的典型例子是 ipairs,它遍历数组的每一个元素,元素的索引需要是数值。
实例(使用函数实现迭代器,实现 n 的平方)
其中,
square
为迭代函数,3
为状态常量,0
为控制变量,square
函数使用3
和0
作为参数调用,之后返回currentNumber
和currentNumber * currentNumber
作为值赋值给k
和v
变量列表,当返回的currentNumber
为nil
时循环结束。(注意:对于 for 循环来说,状态常量3
仅在初始化时获取它的值并传递给迭代函数)
迭代的状态包括被遍历的表(循环过程中不会改变的状态常量)和当前的索引下标(控制变量),ipairs 和迭代函数都很简单,我们在 Lua 中可以这样实现:
当 Lua 调用 ipairs(a) 开始循环时,他获取三个值:迭代函数 iter、状态常量 a、控制变量初始值 0;然后 Lua 调用 iter(a,0) 返回 1, a[1](除非 a[1]=nil);第二次迭代调用 iter(a,1) 返回 2, a[2]……直到第一个 nil 元素。
(3.1.2)多状态迭代器
很多情况下,迭代器需要保存多个状态信息而不是简单的状态常量和控制变量,最简单的方法是使用闭包,还有一种方法就是将所有的状态信息封装到 table 内,将 table 作为迭代器的状态常量,因为这种情况下可以将所有的信息存放在 table 内,所以迭代函数通常不需要第二个参数。
实例:
以上实例中我们可以看到,elementIterator 内使用了闭包函数,实现计算集合大小并输出各个元素。
pairs
和ipairs
异同同
:
都是能遍历集合(表、数组)异
:
ipairs 仅仅遍历值,按照索引升序遍历,索引中断停止遍历。即不能返回 nil,只能返回数字 0,如果遇到 nil 则退出。它只能遍历到集合中出现的第一个不是整数的 key。
pairs 能遍历集合的所有元素。即 pairs 可以遍历集合中所有的 key,并且除了迭代器本身以及遍历表本身还可以返回 nil。