lua基础篇-09

这篇博客详细介绍了Lua中的迭代器概念,包括泛型 for 迭代器的工作原理,无状态和多状态迭代器的差异,并通过实例深入解析了ipairs和pairs的区别。内容涵盖了迭代器的初始化、执行过程以及在不同场景下的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Lua 迭代器

    在Lua中迭代器(iterator)是一种支持指针类型的结构,是一种对象,它能够用来遍历容器中的部分或全部元素,每个迭代器对象代表容器中的确定的地址。

  • 泛型 for 迭代器
  • 无状态的迭代器
  • 多状态的迭代器

泛型 for 迭代器

    泛型 for 在自己内部保存迭代函数,实际上它保存三个值:迭代函数、状态常量、控制变量。
    泛型 for 迭代器提供了集合的 key/value 对,基本语法格式如下:

--代码中,k, v为变量列表;pairs(t)为表达式列表。

for k, v in pairs(t) do
    print(k, v)
end

简单实例如下:

array = {"Lua", "Tutorial"}

for key,value in ipairs(array) 
do
   print(key, value)
end


--以上代码执行输出结果为:
1  Lua
2  Tutorial

以上实例中我们使用了 Lua 默认提供的迭代函数 ipairs(后面再说区别与pairs)。

下面我们看看泛型 for 的执行过程,加深一下对迭代器原理的理解:
    首先,初始化,计算in后面表达式的值,表达式应该返回泛型 for 需要的三个值:迭代函数、状态常量、控制变量;与多值赋值一样,如果表达式返回的结果个数不足三个会自动用nil补足,多出部分会被忽略。

    第二,将状态常量和控制变量作为参数调用迭代函数(注意:对于for结构来说,状态常量没有用处,仅仅在初始化时获取他的值并传递给迭代函数)。

    第三,将迭代函数返回的值赋给变量列表。

    第四,如果返回的第一个值为nil循环结束,否则执行循环体。

    第五,回到第二步再次调用迭代函数


无状态的迭代器

      无状态的迭代器是指不保留任何状态的迭代器,因此在循环中我们可以利用无状态迭代器避免创建闭包花费额外的代价。另外在无状态的迭代器中,只利用状态变量和控制变量这两个值就可以获取到下一个元素。这种无状态迭代器的典型的简单的例子是ipairs,它遍历数组的每一个元素。

以下实例我们使用了一个简单的函数来实现迭代器

function square(iteratorMaxCount,currentNumber)
   if currentNumber<iteratorMaxCount
   then
      currentNumber = currentNumber+1
   return currentNumber, currentNumber*currentNumber
   end
end

for i,n in square,3,0
do
   print(i,n)
end

--以上实例输出结果为:
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
    end
end

function ipairs (a)
    return iter, a, 0
end

当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 = {"Lua", "Tutorial"}

function elementIterator (collection)
   local index = 0
   local count = #collection
   -- 闭包函数
   return function ()
      index = index + 1
      if index <= count
      then
         --  返回迭代器的当前元素
         return collection[index]
      end
   end
end

for element in elementIterator(array)
do
   print(element)
end

--以上实例输出结果为:
Lua
Tutorial

pairs 和 ipairs异同

  • 同:都是能遍历集合(表、数组)
  • 异:ipairs 仅仅遍历值,按照索引升序遍历,索引中断停止遍历。即不能返回 nil,只能返回数字
    0,如果遇到 nil 则退出。它只能遍历到集合中出现的第一个不是整数的 key。 pairs 能遍历集合的所有元素。即 pairs
    可以遍历集合中所有的 key,并且除了迭代器本身以及遍历表本身还可以返回 nil。

    下面我们通过几个实例来理解它们之间的差异:
--实例1:
local tabFiles = {
        [1] = "test2",
        [6] = "test3",
        [4] = "test1"
    }

for k, v in ipairs(tabFiles) do   
    print(k, v) --输出"test2",在key等于2处断开
end


实例2local tabFiles = {
    [2] = "test2",
    [6] = "test3",
    [4] = "test1"
}

for k, v in ipairs(tabFiles) do  
    print(k, v)--[[什么都没输出,为什么?因为控制变量初始值是按升序来遍历的,当key为1时,value为nil,此时便停止了遍历, 所有什么结果都没输出]]--
end


实例3local tabFiles = {
    [2] = "test2",
    [6] = "test3",
    [4] = "test1"
}

for k, v in pairs(tabFiles) do  
    print(k, v)--输出2 test2, 6 test3, 4 test1
end


实例4local tabFiles = {"alpha", "beta", [3] = "no", ["two"] = "yes"} 
for i,v in ipairs(tabFiles ) do  
    print( tabFiles [i] ) --输出前三个   备注:因为第四个key不是整数
end   

for i,v in pairs(tabFiles ) do
    print( tabFiles [i] )    --全部输出 
end
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值