哎,周五晚上我都还这么努力看书,真是好孩子。(小若:不想吐槽了)
其实我都准备玩游戏看电影去的了,但是这书就摆在桌子上,而且正对着我,就想着,扫两眼吧。
结果一扫就不对劲了,因为这内容有点绕,有点小混乱,如果我现在不记录下来的话,下周一可能又要重新看一次了。
好吧,今天我们来聊聊协同程序。
1.什么是协同程序(coroutinue)
大家都知道线程吧?都知道多线程吧?协同程序就和这线程差不多,但是又有比较明显的区别。
多个协同程序在任意时刻只能执行一个,虽然线程在某种意义上也是这样,但这不是一样的概念。
换句话说,一个协同程序在运行的时候,其他协同程序是无法获得执行的机会的。
只有正在运行的协同程序主动挂起时,其他协同程序才有机会执行。
而线程呢?即使不主动休眠,也很有可能因为轮片时间到达而把执行机会让给其他线程。
2.创建协同程序
创建协同程序很简单,咋一看,其实和线程没差别~
代码如下:
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
因为这个协同程序还没有执行完毕,所以只能是挂起状态。
那么,如果让这协同程序继续执行呢?很简单,再次调用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执行之后,协同程序的状态还是挂起呢?不应该是结束了么?结束了就应该是死亡状态了。
而第三次执行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函数下面的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
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.结束
好了,虽然我已经写了这么多了,但是我真正想记录的东西还没开始写呢~!
我了个噗,今晚我还能不能好好玩了…
好吧,内容有点多,下一篇继续…
主要内容:什么是协同(coroutine)?,coroutine_test.lua 文件,实例,生产者-消费者问题,实例什么是协同(coroutine)? Lua 协同程序(coroutine)与线程比较类似:拥有独立的堆栈,独立的局部变量,独立的指令指针,同时又与其它协同程序共享全局变量和其它大部分东西。 协同是非常强大的功能,但是用起来也很复杂。 线程和协同程序区别 线程与协同程序的主要区别在于,一个具有多个线程的程序可以同时运行几个线程,而协同程序却需要彼此协作的运行。 在任一指定时刻只有
什么是协同(coroutine)? Lua 协同程序(coroutine)与线程比较类似:拥有独立的堆栈,独立的局部变量,独立的指令指针,同时又与其它协同程序共享全局变量和其它大部分东西。 协同是非常强大的功能,但是用起来也很复杂。 线程和协同程序区别 线程与协同程序的主要区别在于,一个具有多个线程的程序可以同时运行几个线程,而协同程序却需要彼此协作的运行。 在任一指定时刻只有一个协同程序在运行,
本文向大家介绍举例详解Lua中的协同程序编程,包括了举例详解Lua中的协同程序编程的使用技巧和注意事项,需要的朋友参考一下 协同程序是协同的性质,可以把两个或更多的方法以可控制的方式执行。随着协同程序,在任何给定的时间,只有其协同程序运行之一,这在运行协同程序只能暂停其执行时,明确要求暂停。 上述定义可能看起来模糊。来告诉它更清楚,假设我们有两个方法,一个主程序方法和协同程序。当我们使用恢复功能
本文向大家介绍Lua之协同程序coroutine代码实例,包括了Lua之协同程序coroutine代码实例的使用技巧和注意事项,需要的朋友参考一下 注: resume得到返回值, 如果有对应的yield在wait resume,那么yield的参数作为resum的返回值,第一个返回值表示coroutine没有错误,后面的返回值个数及其值视yeild参数而定。 如果没有yield在wait,那么返回
本文向大家介绍Lua中的协同程序之resume-yield间的数据返回研究,包括了Lua中的协同程序之resume-yield间的数据返回研究的使用技巧和注意事项,需要的朋友参考一下 这次要介绍几个其实很简单,但是一定要小心的返回值规则。 1.resume的参数 resume函数除了第一个参数是协同程序外,还能继续传其他参数,如下代码: resume第二个参数为“resume parame”,这个
本文向大家介绍Lua编程示例(七):协同程序基础逻辑,包括了Lua编程示例(七):协同程序基础逻辑的使用技巧和注意事项,需要的朋友参考一下 输出结果: