关于lua协同程序

Lua 协同程序(coroutine)


什么是协同(coroutine)?

Lua 协同程序(coroutine)与线程比较类似:拥有独立的堆栈,独立的局部变量,独立的指令指针,同时又与其它协同程序共享全局变量和其它大部分东西。

协同是非常强大的功能,但是用起来也很复杂。

  • 基本语法
方法 描述
coroutine.create(fun) 创建并返回一个coroutine, 创建时属于挂起状态, 需要通过coroutine.resume()方法唤醒
coroutine.resume(co,[arg1,...]) 唤醒挂起的coroutine,会返回:唤醒成功或失败与.yield()传入的参数. 如果是运行完毕死亡状态的coroutine则无效,配合conroutine.yield()十分强大
coroutine.yield([arg1,...]) 挂起当前协程, 返回非第一次.resume()传过来的参数
coroutine.status(co) coroutine有三种状态, suspend, running, dead
coroutine.wrap(fun) 返回一个拥有协同程序特性的函数,可以唤醒,可以挂起, 如果这个函数已死亡,再次唤醒的时候会报错
coroutine.running() 返回正在跑的coroutine,一个coroutine就是一个线程,当使用running的时候,就是返回一个corouting的线程号, 在当前协同程序所运行的函数中才有效果
  • 功能演示

.create() + .resume()

创建一个coroutine

local co = coroutine.create(function (str)
  print(str)
  print(str)
end)
coroutine.resume(co) -- 唤醒
--------------
> nil
> nil

刚刚用.create(fun)创建了一个协同程序, 一开始处于挂起状态,需要我们使用.resume(co)它才会运行函数, 然而我们只传入co, 并没有传入额外的参数, 所以打印出来的是~nil

现在创建coroutine并传入参数

local co = coroutine.create(function (str)
  print(str)
  coroutine.yield() -- 挂起
  print(str)
end)
coroutine.resume(co, "帅哥") -- 唤醒
coroutine.resume(co, "美女") -- 唤醒
-------------------------------------
> 帅哥
> 帅哥

在第一次唤醒的时候, 我们把帅哥赋给了str, 所以第一次打印了帅哥, 中间挂起了一次, 可能大家很奇怪, 为什么第二次唤醒,传入了美女, 为什么还是打印帅哥呀?
解析:
我们在第一次传入帅哥 的时候是str接收的, 所以print(str)打印的是帅哥, 然后第二次唤醒的时候, 程序其实是从.yield()之后继续执行, 所以str还是帅哥 , 那么美女又去哪了?, 答案是被.yield()接收了

.yield()

.yield():我可以接收值

local co = coroutine.create(function (str)
  print(str)
  str = coroutine.yield() -- 挂起
  print(str)
end)
coroutine.resume(co, "帅哥") -- 唤醒
coroutine.resume(co, "美女") -- 唤醒
-----------------------------------------
> 帅哥
> 美女

看! 美女是让.yield() 接收了, 当然,你传10个美女它也能一个不落的接收

.yield():我也能传值

local co = coroutine.create(function (str)
  print(str)
  str = coroutine.yield("小姐姐") -- 挂起 并传入一个小姐姐
  print(str)
end)
local stusas, value
stusas, value = coroutine.resume(co, 帅哥) -- true, 小姐姐
stusas, value = coroutine.resume(co, 美女) -- true, nil

.resume(co)是有返回值的:
第一个返回值是一个Boolean类型, 代表唤醒是否成功, 如果成功返回true, 如果协程已dead,则唤醒失败返回false
后面的能数有无值, 要看.yeild()当中是否有传参

.resume().yeild()两个传参关系大概这样:

image.png

.status().running()
status() 有3种状态 suspend挂起, running运行中, dead死亡
running() 只有协程的[.stutas()]为"running"时才有返回值, 否则返回nil

-- 注意 co 不要加local
co = coroutine.create(function (str)
print(coroutine.status(co)) -- running 
print(coroutine.running()) -- thread: 0000000000BBEE90
end)
print(coroutine.status(co)) -- suspended
print(coroutine.running(co)) -- nil

coroutine.resume(co) -- 唤醒

print(coroutine.status(co)) -- dead
print(coroutine.running(co)) -- nil

最后一个.wrap()下次再写

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容