【笨木头Lua专栏】基础补充04:for循环与迭代器的秘密

上一篇我们介绍了,可以使用for循环来完成迭代器的调用,十分简洁。

那么,具体这for循环做了什么呢?我当然没有去看源码,我只是看书而已。

资料来源于《Lua程序设计》第二版,如果这本书的内容没有错的话,那么,本篇文章理论上也不会有错~

笨木头花心贡献,哈?花心?不,是用心~

转载请注明,原文地址: http://www.benmutou.com/archives/1717

文章来源:笨木头与游戏开发

1.返回两个值的迭代器

pairs是能遍历table的key和value的,而我们之前写的dieDaiQi函数只能返回value。

所以,我们要改改dieDaiQi函数,如下:

  1. function dieDaiQi(t)
  2. local i = 0;
  3. return function()
  4. i = i + 1;
  5. if i > #t then
  6. return nil;
  7. end
  8. return i, t[i];
  9. end
  10. end

当然了,这不是一个安全的迭代器,我们假设table中没有nil值。

至于为什么要有一个if i > #t的判断,待会会说到。

使用如下方式调用迭代器:

  1. local t = {"fdsd", "445", "9999"};
  2. for k, v in dieDaiQi(t) do
  3. print(k .. "," .. v);
  4. end

输出结果如下:

[LUA-print] 1,fdsd
[LUA-print] 2,445
[LUA-print] 3,9999

2.for .. in .. do的真面目

【for k, v in dieDaiQi(t) do  end】这段代码实际上等价于以下代码:

  1. do
  2. local _f, _s, _var = dieDaiQi(t);
  3. while true do
  4. local k, v = _f(_s, _var);
  5. _var = k;
  6. if _var == nil then
  7. break;
  8. end
  9. print(k .. "," .. v);
  10. end
  11. end

是不是很复杂?其实它和我们之前第一次调用迭代器的代码很像,我们先删掉复杂的部分,代码变成如下:

  1. do
  2. local _f = dieDaiQi(t);
  3. while true do
  4. local k, v = _f();
  5. if k == nil then
  6. break;
  7. end
  8. print(k .. "," .. v);
  9. end
  10. end

试试运行这段代码,结果如下:

[LUA-print] 1,fdsd
[LUA-print] 2,445
[LUA-print] 3,9999

和直接使用for in循环是一样的结果。

实际上,我说的这些都是废话,因为我们之前就已经说,for in循环就是用来简化迭代器的调用的,所以当然是一样的结果。

3.迭代器函数、恒定状态、控制变量初值

我们来看看for in真面目的第一句代码:local _f, _s, _var = dieDaiQi(t);

三个返回值分别代表迭代器函数(_f)、恒定状态(_s)、控制变量初值(_var)。

迭代器函数:就不用解释了,就是我们的dieDaiQi返回的闭合函数。

恒定状态:其实就是一个变量,这个变量一直不变,所以称之为恒定。

控制变量初值:和恒定相对于的,这是一个会不断改变的变量。

因为我本人没有实际使用过这种特性,所以没法举出实际的例子,只能从理论上去解释。

1.比如我们的dieDaiQi函数,它只有一个返回值,就是那个闭合函数,所以,_s和_var都是nil。

2.接着调用local k, v = _f(_s, _var); 这实际上就是调用了闭合函数,并且将恒定值和变量值都作为参数传递进去。

3.Lua的函数是很自由的,即使_f函数本身没有参数,也可以传参数进去,不会影响什么,所以,两个nil值传进去了,没有任何事情发生,就像是直接调用_f()一样。

4.再下一句代码:_var = k;  这是把闭合函数(_f)的第一个返回值保存起来,因为每次调用闭合函数(_f)返回值都是下一个迭代值,所以_var每次都是不一样的值。

5.如果_var的值为nil,则停止循环,结束迭代。

因此,我们编写迭代器的时候,迭代结束的方式就是让第一个返回值为nil。

那么,如果我们让dieDaiQi函数返回恒定状态和控制变量初值,又是什么样的情况呢?

代码如下:

  1. function dieDaiQi(t)
  2. local i = 0;
  3. return function(s, var)
  4. i = i + 1;
  5. if i > #t then
  6. return nil;
  7. end
  8. print("恒定值=" .. s .. ", 变量值=" .. var)
  9. return i, t[i];
  10. end, 10, 0
  11. end

留意一下,dieDaiQi函数现在会返回三个参数,后面的10和0分别就是恒定状态和控制变量初值。

同时,闭合函数也多了两个参数:s和var。

于是,我们再次用for循环遍历迭代器:

  1. for k, v in dieDaiQi(t) do
  2. print(k .. "," .. v);
  3. end

输出结果如下:

[LUA-print] 恒定值=10, 变量值=0
[LUA-print] 1,fdsd
[LUA-print] 恒定值=10, 变量值=1
[LUA-print] 2,445
[LUA-print] 恒定值=10, 变量值=2
[LUA-print] 3,9999

恒定值自然是一直不变的,而变量值在每一次调用了闭合函数之后,就会赋值为k的值,所以变量值一直按着table的key值在变化。

可能一时有点混乱,不过,只要对照着for .. in .. do .. end对应的实现代码,就很好理解了。

4.结束

终于写完了,我快撑不住了,一晚上写两篇文章,可够折腾的。

现在眼睛都是花的…我不知道我还能坚持多少个晚上…

幸好学习的内容会越来越难,这样我就没法一个晚上就理解透彻,也就没法每晚写一篇教程了~

太好了,呵呵。(小若:想偷懒就偷懒吧,说这么多做什么)

时间: 2024-10-12 13:17:32

【笨木头Lua专栏】基础补充04:for循环与迭代器的秘密的相关文章

【笨木头Lua专栏】基础补充04:迭代器初探

今天学习的内容还蛮有意思的,让我兴奋了一下~ 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址: http://www.benmutou.com/archives/1714 文章来源:笨木头与游戏开发 1.迭代器 什么是迭代器?别傻了,我最讨厌的就是名词解释了,反正就是用来遍历集合的一种方式. 比如,我们最常用的pairs,如下代码: local t = {"fdsd", "445"}; for k, v in pairs(t) do print(&qu

【笨木头Lua专栏】基础补充02:函数的几个特别之处

没想到距离上一篇基础补充已经过了1年多了,近期准备捡回Lua,把基础都补补,今天来聊聊Lua的函数吧~ 0.环境 我突然对Lua又大感兴趣的最主要原因是,Cocos Code IDE開始浮出水面了,它是Cocos2d-x官方出的一款专门针对Cocos2d-x+Lua或JS的IDE.试着用了,尽管不能说非常完美.但,非常值得期待. 所以,本文使用的Lua编辑器就选它了,大家就任意吧~ 笨木头花心贡献,哈?花心?不.是用心~ 转载请注明,原文地址:http://www.benmutou.com/ar

(转载)【笨木头Lua专栏】基础补充20:面向对象——类和继承

终于来了,在Lua中的面向对象编程,相信目前学习Lua的大部分人都是为了开发手机网游吧.而且基本都是奔着脚本语言的热更新特性去的,所以全脚本开发变得十分流行. 对于普及不太广的Lua(相对于C++.Java等主流语言),需要短时间上手开发游戏,对新手而言不算简单.所以大家才更习惯于继续用面向对象思想去折腾Lua吧~ 好了,不唠叨了,我最不喜欢唠叨了.(小若:是是是,你一点都不唠叨,赶紧开讲!) 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址:http://www.benmutou.

(转载)【笨木头Lua专栏】基础补充21:面向对象——多重继承、私密性

在Lua中的多重继承和私密性可能用得比较少,也可能只是我个人用得比较少. 本来想偷懒不写这文章的,因为我今天刚买了个漂移板,连起步都还没学会啊,想多学一会. 咳咳,本着坚持不懈.负责到底的态度,我还是决定随便写几句~(小若:随便写几句是几吨意思啊?!) 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址:http://www.benmutou.com/archives/1800 文章来源:笨木头与游戏开发 1.多重继承之在多个类中查找一个字段 我发现这些高(shen)智(jing)商(

(转载)【笨木头Lua专栏】基础补充22:弱引用table

这次要介绍的内容比较少,就一个--弱引用table 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址:http://www.benmutou.com/archives/1808 文章来源:笨木头与游戏开发 1.无法超越人类智慧的智能--自动内存管理的缺陷 我们都知道,Lua是具备自动内存管理的,好吧,也许有些朋友不知道. 我们只管创建对象,无须删除对象(当然,对于不要的对象你需要设置一下nil值),Lua会自动删除那些被认为是垃圾的对象. 问题就出现在,什么对象才是垃圾对象,有些时候

【笨木头Lua专栏】基础补充06:简单的错误处理

昨天遇到另外一位独立游戏开发者,所以多聊了一会,然后-然后就没有看书了.(小若:借口!借口!) 今天来聊聊错误处理吧,不过毕竟这只是前面的章节,书上的内容似乎有点一笔带过的味道. 没关系,简单更好~ 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址: http://www.benmutou.com/archives/1728 文章来源:笨木头与游戏开发 1.红色警报--error 我们应该能经常看到类似以下的错误信息: [LUA-print] LUA ERROR: [string "

【笨木头Lua专栏】基础补充05:迭代器番外篇

关于迭代器的内容, 还有一点点,不过已经无关紧要了,应该算是一种扩展吧,就一起来开开眼界好了~ 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址: http://www.benmutou.com/archives/1721 文章来源:笨木头与游戏开发 1.避免创建闭合函数 我们之前一直在说的迭代器,都是要创建闭合函数,但,大家有没有想过,有了恒定状态和控制变量之后,是不是就不需要闭合函数了? 先来回顾一下之前的迭代器函数: function dieDaiQi(t) local i =

【笨木头Lua专栏】基础补充07:协同程序初探

哎.周五晚上我都还这么努力看书.真是好孩子.(小若:不想吐槽了) 事实上我都准备rs=1&u=http%3A%2F%2Fwww%2Ebenmutou%2Ecom%2Farchives%2F1733&p=baidu&c=news&n=10&t=tpclicked3_hc&q=25013069_cpr&k=%CD%E6%D3%CE%CF%B7&k0=%B4%F2%D3%A1&kdi0=1&k1=%D7%C0%D7%D3&k

【本·伍德Lua专栏】补充的基础06:简单的错误处理

昨天遇到另外一位独立游戏开发人员,所以多聊了一会,然后-然后就没有看书了.(小若:借口!借口! ) 今天来聊聊错误处理吧.只是毕竟这仅仅是前面的章节.书上的内容似乎有点一笔带过的味道. 没关系,简单更好~ 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址: http://www.benmutou.com/archives/1728 文章来源:笨木头与游戏开发 1.红色警报--error 我们应该能常常看到相似下面的错误信息: [LUA-print] LUA ERROR: [strin