哎,周五晚上我都还这么努力看书,真是好孩子。(小若:不想吐槽了)

其实我都准备玩游戏看电影去的了,但是这书就摆在桌子上,而且正对着我,就想着,扫两眼吧。

结果一扫就不对劲了,因为这内容有点绕,有点小混乱,如果我现在不记录下来的话,下周一可能又要重新看一次了。

好吧,今天我们来聊聊协同程序。

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

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

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

1.什么是协同程序(coroutine)

大家都知道线程吧?都知道多线程吧?协同程序就和这线程差不多,但是又有比较明显的区别。

多个协同程序在任意时刻只能执行一个,虽然线程在某种意义上也是这样,但这不是一样的概念。

换句话说,一个协同程序在运行的时候,其他协同程序是无法获得执行的机会的。

只有正在运行的协同程序主动挂起时,其他协同程序才有机会执行。

而线程呢?即使不主动休眠,也很有可能因为轮片时间到达而把执行机会让给其他线程。

2.创建协同程序

创建协同程序很简单,咋一看,其实和线程没差别~

代码如下:

  1. local co = coroutine.create(function() print("hello coroutine"); end);

协同的程序的操作都在coroutine里,create函数的参数就是协同程序要执行的函数,就这么运行代码是没有效果的。

因为协同程序创建后,默认是挂起状态。

协同程序的四种状态分别为:挂起(suspended)、运行(running)、死亡(dead)、正常(normal)。

要想协同程序运行起来,就要调用resume函数。

如下代码:

  1. local co = coroutine.create(function() print("hello coroutine"); end);
  2. coroutine.resume(co);

输出结果如下:

[LUA-print] hello coroutine

3.更像样的协同程序

刚刚那个协同程序太简陋的,没有任何作用,直接打印一条语句之后就结束了,同时它的状态也变成了死亡状态。

我们来一个帅一点的协同程序:

  1. local co = coroutine.create(function()
  2. for i = 1, 2, 1 do
  3. print("木头挺聪明的+" .. i);
  4. end
  5. end);
  6. coroutine.resume(co);

运行结果如下:

[LUA-print] 木头挺聪明的+1
[LUA-print] 木头挺聪明的+2

所以我就说,电脑就是诚实,这日志打印的,真好看(小若:我们不要理这个神经病了)

4.让协同程序挂起——yield

既然协同程序和线程差不多,那肯定不能让协同程序一次过执行完毕了,这就没有意义了。

我们来看看怎么让协同程序挂起,如下代码:

  1. local co = coroutine.create(function()
  2. for i = 1, 2, 1 do
  3. print("木头挺聪明的+" .. i);
  4. coroutine.yield();
  5. end
  6. end);
  7. coroutine.resume(co);
  8. print(coroutine.status(co));

输出结果如下:

[LUA-print] 木头挺聪明的+1
[LUA-print] suspended

这回就只输出了一条日志就停止了,后面我们还调用了status函数,打印协同程序当前的状态,suspended即为挂起状态。

因为这个协同程序还没有执行完毕,所以只能是挂起状态。

那么,如果让这协同程序继续执行呢?很简单,再次调用resume函数,如代码:

  1. local co = coroutine.create(function()
  2. for i = 1, 2, 1 do
  3. print("木头挺聪明的+" .. i);
  4. coroutine.yield();
  5. end
  6. end);
  7. coroutine.resume(co);
  8. print(coroutine.status(co));
  9. coroutine.resume(co);
  10. print(coroutine.status(co));
  11. coroutine.resume(co);
  12. print(coroutine.status(co));

这次有点复杂了,先看看输出结果:

[LUA-print] 木头挺聪明的+1
[LUA-print] suspended
[LUA-print] 木头挺聪明的+2
[LUA-print] suspended
[LUA-print] dead

我一共执行了三次resume函数,但很显然,这个协同程序的for循环只会执行2次。

那为什么第二次resume执行之后,协同程序的状态还是挂起呢?不应该是结束了么?结束了就应该是死亡状态了。

而第三次执行resume之后,反而没有任何输出,此时的状态才真正切换到死亡状态。

这是为什么呢?(小若:赶紧说,不说我看电影去了)

再来这么看看就明白了,加几条打印代码:

  1. local co = coroutine.create(function()
  2. for i = 1, 2, 1 do
  3. print("木头挺聪明的+" .. i);
  4. coroutine.yield();
  5. print("一次循环结束");
  6. end
  7. print("协同程序结束");
  8. end);
  9. coroutine.resume(co);
  10. print(coroutine.status(co));
  11. coroutine.resume(co);
  12. print(coroutine.status(co));
  13. coroutine.resume(co);
  14. print(coroutine.status(co));

输出结果如下:

[LUA-print] 木头挺聪明的+1
[LUA-print] suspended
[LUA-print] 一次循环结束
[LUA-print] 木头挺聪明的+2
[LUA-print] suspended
[LUA-print] 一次循环结束
[LUA-print] 协同程序结束
[LUA-print] dead

这就很明显了,在协同程序里调用yield函数时,会被挂起,而yield函数的返回要等下一次调用resume函数时才能得到。

所以,yield函数下面的print语句在下一次的resume调用时才被执行。

又所以,当for循环第二次执行时,协同程序被挂起,需要等待再一次resume时,for循环才能真正执行完毕。

这就是这段代码的特殊之处了。

5.resume操作的返回值

其实resume函数是有返回值的。

我们试试运行下面的代码:

  1. local co = coroutine.create(function()
  2. for i = 1, 2, 1 do
  3. coroutine.yield();
  4. end
  5. end);
  6. local result, msg = coroutine.resume(co);
  7. print(result);
  8. print(msg);

输出结果如下:

[LUA-print] true
[LUA-print] nil

resume返回两个值,第一个值代表协同程序是否正常执行,第二个返回值自然是代表错误信息。

我们试试让协同程序出现错误:

  1. local co = coroutine.create(function()
  2. error("呵呵,报错了吧");
  3. end);
  4. local result, msg = coroutine.resume(co);
  5. print(result);
  6. print(msg);

输出结果如下:

[LUA-print] false
[LUA-print] [string "src/main.lua"]:91: 呵呵,报错了吧

6.结束

好了,虽然我已经写了这么多了,但是我真正想记录的东西还没开始写呢~!

我了个噗,今晚我还能不能好好玩了…

好吧,内容有点多,下一篇继续…

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

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

    今天学习的内容还蛮有意思的,让我兴奋了一下~ 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址: http://www.benmutou.com/archives/1714 文章来源:笨木 ...

  2. (转载)【笨木头Lua专栏】基础补充02:函数的几个特别之处

    [笨木头Lua专栏]基础补充02:函数的几个特别之处 笨木头  2014-08-14 21:45   Cocos2d-x Lua   阅读(4,507)   6条评论 没想到距离上一篇基础补充已经过了 ...

  3. (转载)【笨木头Lua专栏】基础补充01:巧说table的几种构造方式

    [笨木头Lua专栏]基础补充01:巧说table的几种构造方式 笨木头  2013-06-4 23:04   Cocos2d-x Lua   阅读(13,556)   13条评论 之前对于Lua的研究 ...

  4. 笨木头Lua专栏 函数的几个特别之处

    [笨木头Lua专栏]基础补充02: [木头Cocos2d-x 029]Lua篇(第04章):来点高难度的,获取Lua表结构数据 只是获取一个全局变量什么的太没意思了,今天我们来玩个高难度的--获取Lu ...

  5. (转载)【笨木头Lua专栏】基础补充07:协同程序初探

    哎,周五晚上我都还这么努力看书,真是好孩子.(小若:不想吐槽了) 其实我都准备玩游戏看电影去的了,但是这书就摆在桌子上,而且正对着我,就想着,扫两眼吧. 结果一扫就不对劲了,因为这内容有点绕,有点小混 ...

  6. 【笨木头Lua专栏】基础补充08:协同程序之resume-yield间的数据返回

    这次要介绍几个事实上非常easy,可是一定要小心的返回值规则. 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明.原文地址: http://www.benmutou.com/archives/173 ...

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

    没想到距离上一篇基础补充已经过了1年多了,最近准备捡回Lua,把基础都补补,今天来聊聊Lua的函数吧~ 0.环境 我突然对Lua又大感兴趣的最主要原因是,Cocos Code IDE开始浮出水面了,它 ...

  8. 【笨木头Lua专栏】基础补充01:巧说table的几种构造方式

    之前对于Lua的研究都是纸上谈兵,没有真正的项目练手,现在公司的项目基本上都是用Lua开发,是时候补充一下我那蹩脚的Lua知识了. 基础数据类型.表达式.循环结构什么的我就不说了,这么简单的东西说了也 ...

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

    关于迭代器的内容, 还有一点点,不过已经无关紧要了,应该算是一种扩展吧,就一起来开开眼界好了~ 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址: http://www.benmutou. ...

最新文章

  1. 华为开发者大会HDC.Cloud技术探秘:云搜索服务技术实践
  2. idea androidx控件不显示预览_如何解决SOLIDWORKS不显示缩略图预览的方法?
  3. Confluence 6 可以自定义的元素
  4. 基于局部均方差相关信息的图像去噪及其在实时磨皮美容算法中的应用。
  5. java.函数参数太多,Java方法参数太多怎么办—Part 2—引入参数对象,javapart
  6. 大象喝水c语言程序,实现大象喝水(c语言)
  7. Eclipse-cdt 配合 gdbserver 进行 arm 程序远程调试
  8. java学习笔记十三
  9. struts2面试问题_Struts2面试问答
  10. 这么奇葩搞笑的代码注释你见过吗
  11. linux core无权限,linux – 为什么编辑core_pattern受限制?
  12. HTML+CSS+JS实现 ❤️响应式的幸运大转盘❤️
  13. 网络基础知识_你家的网络是这么布线的吗?家庭网络布线基础知识普及!
  14. try、catch和finally中都有return语句的情况
  15. 粘性定位(HTML、CSS)
  16. pip更新导致ImportError: cannot import name ‘InvalidSchemeCombination‘ from ‘pip._internal.exceptions‘
  17. 计算机应用用英语,计算机应用常用英语:“windows”
  18. python人口普查数据显示_如何使用FCC的API在Python中查找人口普查数据块并遍历dict列表?...
  19. Word文档调整字间距的方法教程【收藏】
  20. 哪些股票自动交易接口好用呢?

热门文章

  1. Java调用Mysql
  2. 不解,排名靠前那么多的人为什么抄袭我的activit博文??
  3. AFNetworking2.4.1解析
  4. PHP面试技巧——如何应对面试官的“激将法”语言?
  5. Rockland检测开发丨Rockland 免疫分析开发方案
  6. android studio 正式版本
  7. Ubuntu apt安装包 dev、dbg、utils后缀的含义
  8. 2021手游排行前十名分享
  9. PHP读和写Excel文件
  10. 恒大造车,从来都不是梦