协程是轻量级线程,开启协程不等同于开启线程,即不是开启一个新的流水线。通俗点讲,对于一堆任务,通过一个线程轮着对其中每个任务都执行一下。它的厉害之处在于,每运行到一个任务的时候,它都可以从这个任务上一次中断的地方开始运行。这种工作机制类似于操作系统的线程调度。
一、Unity 的 Coroutine 执行顺序
StartCoroutine 函数用于开始一个协同程序,但不是另外开启一个线程,协同程序与 StartCoroutine 函数在同一个线程之内;但是协同程序可以实现“线程切换”的效果,通过 yield 关键字,可以实现在某个事件发生时,线程的控制权回到协同程序。一个协同程序在执行过程中,会在使用了 yield 语句的位置暂停。yield 的返回值控制何时恢复协同程序向下执行。
StartCoroutine 函数在启动协同程序之后,将线程控制权转交给协同程序,协同程序在运行到 yield 语句处(会执行 yield 后的语句),交出控制权给外部线程并等待事件发生,当事件发生时,线程控制权再交回给协同程序。
示例 1:
void Start()
{
print("Starting " +Time.time);----------------------------------------1
StartCoroutine(WaitAndPrint(2));-------------------------------------2
print("Done " +Time.time);-------------------------------------------3
}
IEnumerator WaitAndPrint(float waitTime)
{
yield return new WaitForSeconds(waitTime);------------------------4
print("WaitAndPrint " + Time.time);----------------------------------5
}
代码的执行顺序是:12435
执行到 4 时,协程注册事件,控制权交出给外部线程;外部线程执行3;当协程注册的事件发生时,控制权将由外部线程回到协程,从上一次执行处继续执行。
示例 2:
IEnumerator Start()
{
print("Starting " +Time.time);----------------------------------------1
yield return StartCoroutine(WaitAndPrint(2.0F));------------------------2
print("Done " +Time.time);------------------------------------------3
}
IEnumerator WaitAndPrint(float waitTime)
{
yield return new WaitForSeconds(waitTime);----------------------------4
print("WaitAndPrint " + Time.time);-----------------------------------------5
}
代码的执行顺序是:12453
程序执行到 4,执行 yield return 表达式注册事件,将控制权交出给外部线程,而此时外部线程也要交出控制权,并且执行 yield return 后面的表达式语句,因此会重入 WaitAndPrint 函数,于是接着上次执行位置往下执行,执行语句 5,此时因为 WaitAndPrint 执行完毕,于是将控制权完全交出;之后才执行 3。其中的根本原因是,yield return 不能直接嵌套,后面需要跟一个表达式(事件)。
1.1 unity 中的 yield
yield 后面可以跟的表达式有:
return null
下个 Update 之后恢复
return new WaitForEndOfFrame
下个 OnGUI 之后恢复
return new WaitForFixedUpdate()
下个 FixedUpdate 之后恢复,有可能一帧内多次执行
return new WaitForSeconds(2)
2秒后,等下个Update之后恢复。WaitForSeconds 受 Time.timeScale 影响,当Time.timeScale = 0f 时,yield return new WaitForSecond(x) 将不会恢复
return new WWW(url)
Web请求完成了,Update之后恢复
return StartCorourtine()
新的协成完成了,Update之后恢复
break
退出协程
return Application.LoadLevelAsync(levelName)
load level 完成后恢复,用于异步加载场景
return Resources.UnloadUnusedAssets()
unload 完成后恢复
1.2 规则
返回值必须是 IEnumerator
参数不能加 ref 或 out
函数 Update 和 FixedUpdate 中不能使用 yield 语句,但可以启动协程
yield return 语句不能位于 try-catch 语句块中
yield return 不能放在匿名方法中
yield return 不能放在unsafe语句块中