简单理解Lua 协程(coroutine)

也许更好的阅读体验
协程简单理解为可以暂停的线程,但是同一时刻只有一个协程可以处于运行状态。

coroutine.create()

lua中使用coroutine.create()创建一个协程,参数是一个函数,返回值为创建的协程,这个协程运行内容就是这个函数了。
协程有三种状态挂起、运行、停止。
协程刚创建时处于挂起状态。
可以在代码中使用coroutine.status()查看协程状态。

function foo ()
	print("i'm running ")
end
co = coroutine.create(foo)
print(coroutine.status(co))

运行结果:

suspended

coroutine.resume()

可以在代码中使用coroutine.resume()来唤醒一个被挂起的协程。
运行完毕的协程状态为停止。

function foo ()
	print("i'm running ")
end
co = coroutine.create(foo)
coroutine.resume(co)
print(coroutine.status(co))

运行结果:

i'm running 
dead

coroutine.wrap()

每次都调用coroutine.resume()有点麻烦。可以使用coroutine.wrap()像直接调用函数一样唤醒协程。

function foo ()
	print("i'm running ")
end
co = coroutine.wrap(foo)
co()

运行结果:

i'm running 

coroutine.yield()

可以在函数中使用coroutine.yield()来挂起协程。
下一次resume会从挂起位置继续执行。

function foo ()
	print("i'm running ")
	coroutine.yield()
	print("i' running again")
end
co = coroutine.create(foo)
coroutine.resume(co)
print(coroutine.status(co))
coroutine.resume(co)

运行结果:

i'm running 
suspended
i' running again

coroutine.resume()参数传递

如果函数有参数,第一次调用coroutine.resume()时可以传参。

function say (s)
	print("i say " .. s)
end
co = coroutine.wrap(say)
co("hello")

运行结果:

i say hello

resume和yield之间互换数据

在第一次调用coroutine.resume()后,coroutine.resume()可以和coroutine.yield()互相交换数据。
coroutine.resume()的参数会作为coroutine.yield()函数的结果。
coroutine.yield()的参数会作为coroutine.resume()函数的结果。

function say (s)
	print("i say " .. s)
	print(coroutine.yield("let me have a rest"))
end
co = coroutine.wrap(say)
t = co("hello")
print(t)
co("no")

运行结果:

i say hello
let me have a rest
no

这里say()方法我特意这样写,就是为了让读者直观感受coroutine.yield()直接挂起协程是什么样的。
解释一下,"hello"作为第一次调用co的参数是对应的参数s,运行到coroutine.yield()时挂起协程,并将括号里的"let me have a rest"传递给t,"no"作为第二次调用co的参数时是作为coroutine.yield()的返回值。

posted @ 2024-07-12 15:53  Morning_Glory  阅读(83)  评论(0编辑  收藏  举报
//