lua协程

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

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

local co = coroutine.create(function()
    print("hi)
end)

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

local co = coroutine.create(function()
    print("hi")
end)
print(coroutine.status(co))    --suspended

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

local co = coroutine.create(function()
    print("hi")
end)
print(coroutine.status(co))    --suspended
coroutine.resume(co)    --hi

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

    我们现在来看一下协程函数的yield,该函数可以让一个运行中的协同程序挂起,而之后可以再恢复它的运行。看下面的例子:

local co = coroutine.create(function()
	for i=1,5 do 
		print(i)
		coroutine.yield()
	end
end)

print(coroutine.status(co))
coroutine.resume(co)

print(coroutine.status(co))
coroutine.resume(co)

print(coroutine.status(co))
coroutine.resume(co)
coroutine.resume(co)
coroutine.resume(co)
--coroutine.resume(co)注意这句话
print(coroutine.status(co))
coroutine.resume(co)

   输出:

suspended
1
suspended
2
suspended
3
4
5
suspended    --当把上面的注释打开,这是输出为:dead

    在最后一次调用resume时,协同程序的内容已经执行完毕,并已经返回。因此,这时协同程序处于死亡状态。当在协同程序的执行中发生任何错误,Lua是不会显示错误消息的,而是将执行权返回给resume调用。当coroutine.resume的第一个返回值为false时,就表明协同程序在运行过程中发生了错误;当值为true时,则表明协同程序运行正常。

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

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

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

local co = coroutine.create(function (a, b, c)
     print("co", a, b, c)
end)

coroutine.resume(co, 1, 2, 3)     -- co 1 2 3

    在resume调用返回的内容中,第一个值为 true则表示没有错误,而后面所有的值都是对影yield传入的参数:

local co = coroutine.create(function(a,b)
	coroutine.yield(a+b,a-b)
end)
print(coroutine.resume(co,20,10))

    输出:

true	30	10

    于此对应的时,yield返回的额外值就是对应resume传入的参数:

local co = coroutine.create(function()
	print(coroutine.yield())
end)
coroutine.resume(co)
coroutine.resume(co,4,5)

    输出:

4	5

    最后,当一个协同程序结束时,它的主函数所返回的值都将作为对应resume的返回值:

local co = coroutine.create(function()
	return 6,7
end)
print(coroutine.resume(co))

    输出:

true	6    7

你可能感兴趣的:(lua协程)