【前言】

协同程序与线程差不多,也就是一条执行序列,拥有自己独立的栈、局部变量和指令指针,同时又与其它协同程序共享全局变量和其它大部分东西。从概念上讲,线程与协同程序的主要区别在于,一个具有多个线程的程序可以同时运行几个线程,而协同程序却需要彼此协作的运行。就是说,一个具有多个协同程序的程序在任意时刻只能运行一个协同程序,并且正在运行的协同程序只会在其显式地要求挂起时,它的执行才会暂停。

【协同程序基础】

Lua将所有关于协同程序的函数放置在一个名为“coroutine”的table中。函数create用于创建新的协同程序,它只有一个参数,就是一个函数。该函数的代码就是协同程序需要执行的内容。create会返回一个thread类型的值,用以表示新的协同程序,一般create的参数是一个匿名函数,例如以下代码:

  1. local co = coroutine.create(function () print("Hello WOrld") end)

一个协同程序可以有四种不同的状态:挂起(suspended)、运行(running)、死亡(dead)和正常(normal)。当新创建一个协同程序时,它处于挂起状态,言外之意就是,协同程序不会在创建它时自动执行其内容,我们可以通过函数status来检查协同程序的状态。

  1. local co = coroutine.create(function () print("Hello WOrld") end)
  2. print(coroutine.status(co)) -- suspended

函数coroutine.resume用于启动或再次启动一个协同程序的执行,并将其状态由挂起改为运行:

  1. local co = coroutine.create(function () print("Hello WOrld") end)
  2. print(coroutine.status(co)) -- suspended
  3. coroutine.resume(co) -- Hello World

上面的代码中,我调用了resume函数,将协同程序co由suspended改为running状态,当打印了Hello World之后,协同程序co就处于死亡状态。

到目前为止,协同程序就是一种函数调用。其实,协同程序的真正强大之处在于函数yield的使用上,该函数可以让一个运行中的协同程序挂起,而之后可以再恢复它的运行,例如以下代码:

  1. local co = coroutine.create(function ()
  2. for i = , do
  3. print("co", i)
  4. coroutine.yield()
  5. end
  6. end)
  7.  
  8. -- 打印初始状态
  9. print(coroutine.status(co)) -- suspended
  10.  
  11. -- 唤醒协同程序co
  12. coroutine.resume(co) -- 打印co 1
  13.  
  14. -- 打印协同程序的状态
  15. print(coroutine.status(co)) -- suspended
  16.  
  17. -- 再次唤醒协同程序co
  18. coroutine.resume(co) -- 打印co 2
  19.  
  20. -- 打印协同程序的状态
  21. print(coroutine.status(co)) -- suspended
  22.  
  23. coroutine.resume(co) -- 打印co 3
  24. coroutine.resume(co) -- 打印co 4
  25. coroutine.resume(co) -- 打印co 5
  26. coroutine.resume(co) -- 打印co 6
  27. coroutine.resume(co) -- 打印co 7
  28. coroutine.resume(co) -- 打印co 8
  29. coroutine.resume(co) -- 打印co 9
  30. coroutine.resume(co) -- 打印co 10
  31. coroutine.resume(co) -- 什么都不打印
  32. print(coroutine.status(co)) -- dead
  33. coroutine.resume(co)

当在协同程序的执行中发生任何错误,Lua是不会显示错误消息的,而是将执行权返回给resume调用。当coroutine.resume的第一个返回值为false时,就表明协同程序在运行过程中发生了错误;当值为true时,则表明协同程序运行正常。

当一个协同程序A唤醒另一个协同程序B时,协同程序A就处于一个特殊状态,既不是挂起状态(无法继续A的执行),也不是运行状态(是B在运行)。所以将这时的状态称为“正常”状态。

Lua的协同程序还具有一项有用的机制,就是可以通过一对resume-yield来交换数据。在第一次调用resume时,并没有对应的yield在等待它,因此所有传递给resume的额外参数都视为协同程序主函数的参数。如下述代码:

当协同程序中没有yield时,第一次调用resume,所有传递给resume的额外参数都将视为协同程序主函数的参数,如以下代码:

  1. local co = coroutine.create(function (a, b, c)
  2. print("co", a, b, c)
  3. end)
  4.  
  5. coroutine.resume(co, , , ) -- co 1 2 3

当协同程序中存在yield时,一切就变的复杂了,先来分析一下这个流程:

  1. 调用resume,将协同程序唤醒;
  2. 协同程序运行;
  3. 运行到yield语句;
  4. yield挂起协同程序,第一次resume返回;(注意:此处yield返回,参数是resume的参数)
  5. 第二次resume,再次唤醒协同程序;(注意:此处resume的参数中,除了第一个参数,剩下的参数将作为yield的参数)
  6. yield返回;
  7. 协同程序继续运行;

此处从其它博客中借鉴的一部分代码,可以说明上面的调用流程:

  1. function foo (a)
  2. print("foo", a) -- foo 2
  3. return coroutine.yield( * a) -- return 2 * a
  4. end
  5.  
  6. co = coroutine.create(function (a , b)
  7. print("co-body", a, b) -- co-body 1 10
  8. local r = foo(a + )
  9.  
  10. print("co-body2", r)
  11. local r, s = coroutine.yield(a + b, a - b)
  12.  
  13. print("co-body3", r, s)
  14. return b, "end"
  15. end)
  16.  
  17. print("main", coroutine.resume(co, , )) -- true, 4
  18. print("------")
  19. print("main", coroutine.resume(co, "r")) -- true 11 -9
  20. print("------")
  21. print("main", coroutine.resume(co, "x", "y")) -- true 10 end
  22. print("------")
  23. print("main", coroutine.resume(co, "x", "y")) -- false cannot resume dead coroutine
  24. print("------")

输出结果如下:

  1. >lua -e "io.stdout:setvbuf 'no'" "test.lua"
  2. co-body
  3. foo
  4. main true
  5. ------
  6. co-body2 r
  7. main true -
  8. ------
  9. co-body3 x y
  10. main true end
  11. ------
  12. main false cannot resume dead coroutine
  13. ------
  14. >Exit code:

resume和yield的配合强大之处在于,resume处于主程中,它将外部状态(数据)传入到协同程序内部;而yield则将内部的状态(数据)返回到主程中。

【生产者-消费者问题】

现在我就使用Lua的协同程序来完成生产者-消费者这一经典问题。生产者生产东西,消费者消费生产者生产的东西。

  1. local newProductor
  2.  
  3. function productor()
  4. local i =
  5. while true do
  6. i = i +
  7. send(i) -- 将生产的物品发送给消费者
  8. end
  9. end
  10.  
  11. function consumer()
  12. while true do
  13. local i = receive() -- 从生产者那里得到物品
  14. print(i)
  15. end
  16. end
  17.  
  18. function receive()
  19. local status, value = coroutine.resume(newProductor)
  20. return value
  21. end
  22.  
  23. function send(x)
  24. coroutine.yield(x) -- x表示需要发送的值,值返回以后,就挂起该协同程序
  25. end
  26.  
  27. -- 启动程序
  28. newProductor = coroutine.create(productor)
  29. consumer()

Lua中的协同程序的更多相关文章

  1. Lua中的协同程序 coroutine

    Lua中的协程和多线程很相似,每一个协程有自己的堆栈,自己的局部变量,可以通过yield-resume实现在协程间的切换.不同之处是:Lua协程是非抢占式的多线程,必须手动在不同的协程间切换,且同一时 ...

  2. Lua中的协同程序 coroutine(转)

    Lua中的协程和多线程很相似,每一个协程有自己的堆栈,自己的局部变量,可以通过yield-resume实现在协程间的切换.不同之处是:Lua协程是非抢占式的多线程,必须手动在不同的协程间切换,且同一时 ...

  3. 【转】Unity中的协同程序-使用Promise进行封装(三)

    原文:http://gad.qq.com/program/translateview/7170967 译者:崔国军(飞扬971)    审校:王磊(未来的未来) 在这个系列的最后一部分文章,我们要通过 ...

  4. 【转】Unity中的协同程序-使用Promise进行封装(二)

    原文:http://gad.qq.com/program/translateview/7170970 译者:王磊(未来的未来)    审校:崔国军(飞扬971)   在上一篇文章中,我们的注意力主要是 ...

  5. 【转】Unity中的协同程序-使用Promise进行封装(一)

    原文:http://gad.qq.com/program/translateview/7170767 译者:陈敬凤(nunu)    审校:王磊(未来的未来) 每个Unity的开发者应该都对协同程序非 ...

  6. Unity 中的协同程序

    今天咱就说说,协同程序coroutine.(这文章是在网吧敲的,没有unity,但是所有结论都被跑过,不管你信得过我还是信不过我,都要自己跑一下看看,同时欢迎纠错)先说说啥是协程:协同程序是一个非常让 ...

  7. 《Lua程序设计》9.1 协同程序基础 学习笔记

    协同程序(coroutine)与线程(thread)差不多,也就是一条执行序列,拥有自己独立的栈.局部变量和指令指针,同时又与其他协同程序共享全局变量和其他大部分东西.从概念上讲线程与协同程序的主要区 ...

  8. Unity3D协同程序(Coroutine)

    摘要下: 1. coroutine, 中文翻译"协程".这个概念可能有点冷门,不过百度之,说是一种很古老的编程模型了,以前的操作系统里进程调度里用到过,现在操作系统的进程调度都是根 ...

  9. 【转】关于Unity协同程序(Coroutine)的全面解析

    http://www.unity.5helpyou.com/2658.html 本篇文章我们学习下unity3d中协程Coroutine的的原理及使用 1.什么是协调程序 unity协程是一个能暂停执 ...

随机推荐

  1. SQL分组求每组最大值问题的解决方法收集 (转载)

    例如有一个表student,其结构如下: id      name     sort      score 1        张三      语文      82 2        李四      数 ...

  2. Jmeter名词注解

    取值 ${ip}排除 .*\.js .*\.css .*\.png .*\.gif .*\.msp .*\.js 提取值 (.+?) (.*?)[() 括起来的部分就是需要提取的,对于你要提的内容需要 ...

  3. 关于学习springboot和springcloud的很不错的教程

    近日,逐步开始学习了springboot和springcloud.本以为很简单,但是随着学习的深入,发现其中有很多地方都需要认真揣摩.凡事都需要循序渐进,有一个好的开端就是成功的一半.于是在浩瀚的网络 ...

  4. MyBatis的接口式编程Demo

    很久没细看过MyBatis了,时间一长就容易忘记. 下面是一个接口式编程的例子. 这里的例子一共分为4步: 1 首先要有一个namespace为接口的全类名的映射文件,该例中是 IMyUser.xml ...

  5. 其它综合-企业级CentOS 7.6 操作系统的安装

    企业级CentOS 7.6版本安装过程 1. 环境: 使用的虚拟机软件是VMware,版本为 12 .(网上一搜一大推,在此不再演示.) 使用的ISO镜像为CentOS7.6.(自己也可以在网上搜镜像 ...

  6. Datatable get请求传参应用

    以关注页面为例: html: <div class="row"> <div class="col-md-12 col-sm-12 col-xs-12&q ...

  7. Python future使用

    Python的每个新版本都会增加一些新的功能,或者对原来的功能作一些改动.有些改动是不兼容旧版本的,也就是在当前版本运行正常的代码,到下一个版本运行就可能不正常了. 从Python 2.7到Pytho ...

  8. linux 安装所有软件可以使用这个网站搜索RPM包

    #很方便很实用  强烈推荐 https://pkgs.org/

  9. 【ARC101F】Robots and Exits 树状数组

    题目大意 有 \(n\) 个机器人和 \(m\) 个出口. 这 \(n\) 个机器人的初始位置是 \(a_1,a_2,\ldots,a_n\),这 \(m\) 个出口的位置是 \(b_1,b_2,\l ...

  10. Python 面向对象高阶-----metaclass

    Python 面向对象高阶-----metaclass 前言 类也是对象,既然类是对象,那就自然是某个东西的实例化,这个东西就是type 首先看下type是怎么回事 type type最常用的方法就是 ...