迭代器(iterator)是一种对象,它能够用来遍历标准模板库容器中的部分或全部元素,每个迭代器对象代表容器中的确定的地址。
在 Lua 中迭代器是一种支持指针类型的结构,它可以遍历集合的每一个元素。
泛型 for 在自己内部保存迭代函数,实际上它保存三个值:迭代函数、状态常量、控制变量。
泛型 for 迭代器提供了集合的 key/value 对,语法格式如下:
for k, v in pairs(t) do print(k, v)上面代码中,k, v为变量列表;pairs(t)为表达式列表。
查看以下实例:
array = { "Google" , "Runoob" }以上代码执行输出结果为:
1 Google 2 Runoob下面我们看看泛型 for 的执行过程:
首先,初始化,计算 in 后面表达式的值,表达式应该返回泛型 for 需要的三个值:迭代函数、状态常量、控制变量;与多值赋值一样,如果表达式返回的结果个数不足三个会自动用 nil 补足,多出部分会被忽略。在Lua中我们常常使用函数来描述迭代器,每次调用该函数就返回集合的下一个元素。Lua 的迭代器包含以下两种类型:
无状态的迭代器是指不保留任何状态的迭代器,因此在循环中我们可以利用无状态迭代器避免创建闭包花费额外的代价。
每一次迭代,迭代函数都是用两个变量(状态常量和控制变量)的值作为参数被调用,一个无状态的迭代器只利用这两个值可以获取下一个元素。
这种无状态迭代器的典型的简单的例子是 ipairs,它遍历数组的每一个元素,元素的索引需要是数值。
以下实例我们使用了一个简单的函数来实现迭代器,实现 数字 n 的平方:
function square ( iteratorMaxCount , currentNumber )以上实例输出结果为:
1 1 2 4 3 9
迭代的状态包括被遍历的表(循环过程中不会改变的状态常量)和当前的索引下标(控制变量),ipairs 和迭代函数都很简单,我们在 Lua 中可以这样实现:
function
iter
(
a
,
i
)
i
=
i
+
1
local
v
=
a
[
i
]
if
v
then
return
i
,
v
function
ipairs
(
a
)
return
iter
,
a
,
0
当 Lua 调用 ipairs(a) 开始循环时,他获取三个值:迭代函数 iter、状态常量 a、控制变量初始值 0;然后 Lua 调用 iter(a,0) 返回 1, a[1](除非 a[1]=nil);第二次迭代调用 iter(a,1) 返回 2, a[2]……直到第一个 nil 元素。
很多情况下,迭代器需要保存多个状态信息而不是简单的状态常量和控制变量,最简单的方法是使用闭包,还有一种方法就是将所有的状态信息封装到 table 内,将 table 作为迭代器的状态常量,因为这种情况下可以将所有的信息存放在 table 内,所以迭代函数通常不需要第二个参数。
以下实例我们创建了自己的迭代器:
array = { "Google" , "Runoob" }以上实例输出结果为:
Google Runoob以上实例中我们可以看到,elementIterator 内使用了闭包函数,实现计算集合大小并输出各个元素。
无风
wuf***2@126.com
SllyLi
527***346@qq.com
同:都是能遍历集合(表、数组)
异:ipairs 仅仅遍历值,按照索引升序遍历,索引中断停止遍历。即不能返回 nil,只能返回数字 0,如果遇到 nil 则退出。它只能遍历到集合中出现的第一个不是整数的 key。 pairs 能遍历集合的所有元素。即 pairs 可以遍历集合中所有的 key,并且除了迭代器本身以及遍历表本身还可以返回 nil。
local tabFiles = { [1] = "test2", [6] = "test3", [4] = "test1" for k, v in ipairs(tabFiles) do --输出"test2",在key等于2处断开 print(k, v) local tabFiles = { [2] = "test2", [6] = "test3", [4] = "test1" for k, v in ipairs(tabFiles) do --[[什么都没输出,为什么?因为控制变量初始值是按升序来遍历的,当key为1时,value为nil,此时便停止了遍历, 所有什么结果都没输出]]-- print(k, v) local tabFiles = { [2] = "test2", [6] = "test3", [4] = "test1" for k, v in pairs(tabFiles) do --输出2 test2, 6 test3, 4 test1 print(k, v) local tabFiles = {"alpha", "beta", [3] = "no", ["two"] = "yes"} for i,v in ipairs(tabFiles ) do --输出前三个 备注:因为第四个key不是整数 print( tabFiles [i] ) for i,v in pairs(tabFiles ) do --全部输出 print( tabFiles [i] )道言空
276***597@qq.com
执行输出结果为:
1 php欢迎指教讨论。
kimomi
dol***xie@163.com
泛型 for 在迭代的时候每次调用的是闭包函数,迭代函数只是开始的时候调用一次。
function eleiter(t) local index = 0 print('in eleiter function') --> 每次调用迭代函数都说一句:in eleiter function return function() print('I am here.') --> 每次调用闭包函数都说一句:I am here index = index + 1 return t[index] t = {'one','two','three','four','five'} for ele in eleiter(t) do print(ele)in eleiter function I am here. I am here. I am here. three I am here. I am here. I am here.
asdf
asd***sdf.com
div2009
div***9@foxmail.com
for k, v in pairs(t) do print(k, v)泛 for 中k,v的作用,k作为变量存放索引,v作为变量存放索引对应的值。类似于C语言中的数组索引和数值值的概念。(C语音:a[1]="a",1是索引,"a"是值)。
不论是pairs 还是ipairs都是顺序遍历 t 其中索引存放在k中,对应值放在v中,从1开始。
不同的是pairs会遍历所有输出,ipairs遇到nil值中止。下面例子中我用"a"~"d"作为索引,pairs会全部输出。
ipairs会在一开始,因为t[1]值为nil的情况中止输出。
local t={} t["a"]=1 t["b"]=2 t["c"]=3 t["d"]=4 print("pairs:") -- 以pairs输出 for k,v in pairs(t) do print(k .." , ".. v) print("ipairs:") -- 以ipairs输出 for g,b in ipairs(t) do print(g .." , ".. b)pairs: a , 1 d , 4 c , 3 b , 2 ipairs:
Ingnary
ing***yk@outlook.com
tab = {1,2} tab["key"]=6 for i,v in ipairs(tab) do print(i.." "..v) print(tab["key"]) print("----------------------------") for i,v in pairs(tab) do print(i.." "..v) print(tab["key"]) ---------------------------- key 6 6
自强不息的sun
694***050@qq.com
笔记上的 iter 实现是有 bug 的,和实际的不一样,判断标准永远是 nil 才退出循环,而它漏掉了 false。
可以看我的示例代码就可以发现异同了:
array = {"Google", "Runoob"} for key,value in ipairs(array) do print(key, value) if #array < 4 then array[#array + 1] = nil elseif #array < 8 then array[#array + 1] = '2'上面这样做的话会得到:
1 Google 2 Runoob如果将 nil 修改成 false:
array = {"Google", "Runoob"} for key,value in ipairs(array) do print(key, value) if #array < 4 then array[#array + 1] = false elseif #array < 8 then array[#array + 1] = '2'则会得到:
1 Google 2 Runoob 3 false 4 false 5 2 6 2 7 2 8 2