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()
两个传参关系大概这样:
.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