【笨木头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&kdi1=1&k2=%CD%E6%D3%CE%CF%B7&kdi2=1&k3=%D0%AD%CD%AC&kdi3=1&sid=83c2c7ba4db8ba0b&ch=0&tu=u1682305&jk=feb478688750bc12&cf=29&fv=15&stid=9&urlid=0&luki=3&seller_id=1&di=128" target="_blank" mpid="6" style="border: 0px; margin: 0px; padding: 0px; color: rgb(0, 136, 221); text-decoration: none;">玩游戏
结果一扫就不正确劲了,由于这内容有点绕,有点小混乱,假设我如今不记录下来的话。下周一可能又要又一次看一次了。
好吧,今天我们来聊聊协同程序。
笨木头花心贡献,哈?花心?不,是用心~
转载请注明,原文地址: http://www.benmutou.com/archives/1733
文章来源:笨木头与游戏开发
1.什么是协同程序(coroutine)
大家都知道线程吧?都知道多线程吧?协同程序就和这线程差点儿相同,可是又有比較明显的区别。
多个协同程序在随意时刻仅仅能运行一个,尽管线程在某种意义上也是这样,但这不是一样的概念。
换句话说,一个协同程序在运行的时候。其它协同程序是无法获得运行的机会的。
仅仅有正在运行的协同程序主动挂起时,其它协同程序才有机会运行。
而线程呢?即使不主动休眠。也非常有可能由于轮片时间到达而把运行机会让给其它线程。
2.创建协同程序
创建协同程序非常easy。咋一看。事实上和线程没区别~
代码例如以下:
- local co = coroutine.create(function() print("hello coroutine"); end);
协同的程序的操作都在coroutine里,create函数的參数就是协同程序要运行的函数,就这么运行代码是没有效果的。
由于协同程序创建后,默认是挂起状态。
协同程序的四种状态分别为:挂起(suspended)、运行(running)、死亡(dead)、正常(normal)。
要想协同程序运行起来,就要调用resume函数。
例如以下代码:
- local co = coroutine.create(function() print("hello coroutine"); end);
- coroutine.resume(co);
输出结果例如以下:
[LUA-print] hello coroutine
3.更像样的协同程序
刚刚那个协同程序太简陋的。没有不论什么作用,直接打印一条语句之后就结束了,同一时候它的状态也变成了死亡状态。
我们来一个帅一点的协同程序:
- local co = coroutine.create(function()
- for i = 1, 2, 1 do
- print("木头挺聪明的+" .. i);
- end
- end);
- coroutine.resume(co);
运行结果例如以下:
[LUA-print] 木头挺聪明的+1
[LUA-print] 木头挺聪明的+2
所以我就说,电脑就是诚实。这日志打印的,真好看(小若:我们不要理这个神经病了)
4.让协同程序挂起——yield
既然协同程序和线程差点儿相同。那肯定不能让协同程序一次过运行完成了,这就没有意义了。
我们来看看怎么让协同程序挂起,例如以下代码:
- local co = coroutine.create(function()
- for i = 1, 2, 1 do
- print("木头挺聪明的+" .. i);
- coroutine.yield();
- end
- end);
- coroutine.resume(co);
- print(coroutine.status(co));
输出结果例如以下:
[LUA-print] 木头挺聪明的+1
[LUA-print] suspended
这回就仅仅输出了一条日志就停止了,后面我们还调用了status函数,打印协同程序当前的状态,suspended即为挂起状态。
由于这个协同程序还没有运行完成。所以仅仅能是挂起状态。
那么,假设让这协同程序继续运行呢?非常easy,再次调用resume函数。如代码:
- local co = coroutine.create(function()
- for i = 1, 2, 1 do
- print("木头挺聪明的+" .. i);
- coroutine.yield();
- end
- end);
- coroutine.resume(co);
- print(coroutine.status(co));
- coroutine.resume(co);
- print(coroutine.status(co));
- coroutine.resume(co);
- print(coroutine.status(co));
这次有点复杂了。先看看输出结果:
[LUA-print] 木头挺聪明的+1
[LUA-print] suspended
[LUA-print] 木头挺聪明的+2
[LUA-print] suspended
[LUA-print] dead
我一共运行了三次resume函数。但非常显然,这个协同程序的for循环仅仅会运行2次。
那为什么第二次resume运行之后,协同程序的状态还是挂起呢?不应该是结束了么?结束了就应该是死亡状态了。
而第三次运行resume之后,反而没有不论什么输出,此时的状态才真正切换到死亡状态。
这是为什么呢?(小若:赶紧说,不说我看电影去了)
- local co = coroutine.create(function()
- for i = 1, 2, 1 do
- print("木头挺聪明的+" .. i);
- coroutine.yield();
- print("一次循环结束");
- end
- print("协同程序结束");
- end);
- coroutine.resume(co);
- print(coroutine.status(co));
- coroutine.resume(co);
- print(coroutine.status(co));
- coroutine.resume(co);
- 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函数是有返回值的。
我们试试运行以下的代码:
- local co = coroutine.create(function()
- for i = 1, 2, 1 do
- coroutine.yield();
- end
- end);
- local result, msg = coroutine.resume(co);
- print(result);
- print(msg);
输出结果例如以下:
[LUA-print] true
[LUA-print] nil
resume返回两个值,第一个值代表协同程序是否正常运行。第二个返回值自然是代表错误信息。
我们试试让协同程序出现错误:
- local co = coroutine.create(function()
- error("呵呵,报错了吧");
- end);
- local result, msg = coroutine.resume(co);
- print(result);
- print(msg);
输出结果例如以下:
[LUA-print] false
[LUA-print] [string "src/main.lua"]:91: 呵呵。报错了吧
6.结束
好了,尽管我已经写了这么多了,可是我真正想记录的东西还没開始写呢~!
我了个噗,今晚我还能不能好好玩了…
好吧,内容有点多,下一篇继续…
【笨木头Lua专栏】基础补充07:协同程序初探的更多相关文章
- 【笨木头Lua专栏】基础补充04:迭代器初探
今天学习的内容还蛮有意思的,让我兴奋了一下~ 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明,原文地址: http://www.benmutou.com/archives/1714 文章来源:笨木 ...
- 【笨木头Lua专栏】基础补充08:协同程序之resume-yield间的数据返回
这次要介绍几个事实上非常easy,可是一定要小心的返回值规则. 笨木头花心贡献,哈?花心?不,是用心~ 转载请注明.原文地址: http://www.benmutou.com/archives/173 ...
- 【笨木头Lua专栏】基础补充02:函数的几个特别之处
没想到距离上一篇基础补充已经过了1年多了,近期准备捡回Lua,把基础都补补,今天来聊聊Lua的函数吧~ 0.环境 我突然对Lua又大感兴趣的最主要原因是,Cocos Code IDE開始浮出水面了,它 ...
- 【笨木头Lua专栏】基础补充05:迭代器番外篇
关于迭代器的内容, 另一点点,只是已经无关紧要了.应该算是一种扩展吧.就一起来开开眼界好了~ 笨木头花心贡献.哈?花心?不.是用心~ 转载请注明,原文地址: http://www.benmutou.c ...
- Lua学习笔记4. coroutine协同程序和文件I/O、错误处理
Lua学习笔记4. coroutine协同程序和文件I/O.错误处理 coroutine Lua 的协同程序coroutine和线程比较类似,有独立的堆栈.局部变量.独立的指针指令,同时又能共享全局变 ...
- 【本·伍德Lua专栏】补充的基础09:使用table.concat将一个大的字符串
近期2天都没有写新的文章了.主要是近期的内容没有特别有意思的. 之前的协同程序也临时没有感觉到特别适用的地方.今天在看数据结构的部分,也是没多大意思(不代表没用). 但是突然发现了一个有意思的地方,那 ...
- 【本·伍德Lua专栏】补充的基础06:简单的错误处理
昨天遇到另外一位独立游戏开发人员,所以多聊了一会,然后-然后就没有看书了.(小若:借口!借口! ) 今天来聊聊错误处理吧.只是毕竟这仅仅是前面的章节.书上的内容似乎有点一笔带过的味道. 没关系,简单更 ...
- 《Lua程序设计》9.1 协同程序基础 学习笔记
协同程序(coroutine)与线程(thread)差不多,也就是一条执行序列,拥有自己独立的栈.局部变量和指令指针,同时又与其他协同程序共享全局变量和其他大部分东西.从概念上讲线程与协同程序的主要区 ...
- Lua的协同程序初探
Content: 前两天把Lua的协同程序概念看了一下,不是很懂,只能说<Programming In Lua>中把它解释成线程让人很好的理解起来,但是真正去看的时候,收获并不是很大.第一 ...
随机推荐
- java 定时任务-servlet
在web.xml中配置监听类. <listener> <listener-class>com.skin.generate.action.TaskManager</list ...
- 洛谷——P1640 [SCOI2010]连续攻击游戏
P1640 [SCOI2010]连续攻击游戏 题目描述 lxhgww最近迷上了一款游戏,在游戏里,他拥有很多的装备,每种装备都有2个属性,这些属性的值用[1,10000]之间的数表示.当他使用某种装备 ...
- 为何ARM linux会引入Device Tree(转)
http://www.360doc.com/content/14/0522/20/14530056_380011180.shtml
- java线程总结--synchronized关键字,原理以及相关的锁
在多线程编程中,synchronized关键字非常常见,当我们需要进行“同步”操作时,我们很多时候需要该该关键字对代码块或者方法进行锁定.被synchronized锁定的代码块,只能同时有一条线程访问 ...
- [第一波模拟\day3\T3]{益智游戏}(game.cpp)
[问题描述] 小P和小R在玩一款益智游戏.游戏在一个正权有向图上进行. 小P控制的角色要从A点走最短路到B点,小R控制的角色要从C点走最短路到D点. 一个玩家每回合可以有两种选择,移动到一个相邻节点或 ...
- 初识Web框架
一.Web框架本质 Python的Web框架分为两类: 通过socket自己写程序,自己处理请求: 基于Wsgi(Web Server Gateway Interface:Web服务网关接口),处理请 ...
- js总结(三):面向对象,prototype ,oo模拟
http://aralejs.org/class/docs/competitors.html http://javascript.crockford.com/prototypal.html proto ...
- 服务器端架构及实战 — C#分享
简介 此文是我2008年读研究生期间做的C#项目技术分享,给计算机专业学生的一些经验分享. 当时工作2年后读研. 计算机基础了解及介绍 了解计算机的核心课程和大家的理解 二进制的历史和原理 数字逻辑及 ...
- [codeforces538E]Demiurges Play Again
[codeforces538E]Demiurges Play Again 试题描述 Demiurges Shambambukli and Mazukta love to watch the games ...
- 兴奋剂检查(vijos 1426)
背景 北京奥运会开幕了,这是中国人的骄傲和自豪,中国健儿在运动场上已经创造了一个又一个辉煌,super pig也不例外……………… 描述 虽然兴奋剂是奥运会及其他重要比赛的禁药,是禁止服用的.但是运动 ...