admin 管理员组

文章数量: 887021

[Unity3D]

本文是个人对Unity协程的一些理解和总结.Unity协程长的有点像线程,但却不是线程.因为协程仍然是在主线程中执行,且在使用时不用考虑同步与锁的问题.协程只是控制代码等到特定的时机后再执行后续步骤.


启动协程


Unity 5.x中使用StartCoroutine方法开启协程,其方式有以下几种.

//形式一
StartCoroutine(CustomCorutineFn());
StartCoroutine(CustomCorutineFn(7));//向方法中传递参数
//形式二
StartCoroutine(“CustomCorutineFn”);
StartCoroutine(“CustomCorutineFn”,7);//向方法中传递参数

以上两种形式都可开起当前MonoBehaviour对象的协程,注意:当你想停止MonoBehaviour对象中的某个协程时,开启与停止协程需要使用相同的形式,不可混合使用.
协程中有多种等待方式,例如:等到FixedUpdate结束后才执行,代码如下.

   bool canExcute = true;void FixedUpdate(){if (canExcute){Debug.Log("FixedUpdate" );}}IEnumerator Corutine_WaitForFixedUpdate(){yield return new WaitForFixedUpdate();Debug.Log(string .Format("====>{0}    time:{1}", 1, Time .time));yield return new WaitForFixedUpdate();Debug.Log(string .Format("====>{0}    time:{1}", 2, Time .time));}

输出结果如下.

官方文档Monobehaviour的函数执行顺序图,就对协程再次执行的时机做了很好的描述.

以上只是一个示意图,详细信息请看官方文档.
链接.html

yield null:协程将在下一帧所有脚本的Update执行之后,再继续执行.
yield WaitForSeconds:协程在延迟指定时间,且当前帧所有脚本的 Update全都执行结束后才继续执行.
yield WaitForFixedUpdate:协程在所有脚本的FixedUpdate执行之后,再继续执行.
yield WWW:协程在WWW下载资源完成后,再继续执行.
yield StartCoroutine:协程在指定协程执行结束后,再继续执行.
WaitForSecondsRealtime:与WaitForSeconds类似,但不受时间缩放影响.
WaitWhile:当返回条件为假时才执行后续步骤.

协程的执行也会受到其他因素的影响,例如:当时间缩放值Time.timeScale被修改后,放大或者缩小.FixedUpdate 方法会受影响,则WaitForFixedUpdate也会跟着受影响;当Update方法中同步加载较大的对象时,WaitForSeconds所指定的时间就可能会与实际的时间不一致.所以在执行协程等待时,要视情况而定.

多个gameObject对象开启协程,执行顺序又是如何呢?
假如场景中存在A,B两个gameObject对象,均使用WaitForFixedUpdate方式等待,则等待执行的部分,会在A,B两个对象的FixedUpdate都执行结束后,才开始执行当前帧后续可执行的部分.源码如下:

   void Awake(){StartCoroutine(Corutine_WaitForFixedUpdate());}IEnumerator Corutine_WaitForFixedUpdate(){Debug.Log(string.Format("{1} : {0}", 0, name));yield return new WaitForFixedUpdate();Debug.Log(string.Format("{1} : {0}", 1, name));}bool canExcute = false;void FixedUpdate(){if (!canExcute){canExcute = true;Debug.Log(name + " FixedUpdate");}}

执行后,输出结果入下:


停止协程


在开发中可能会开启多个协程,如果你想停止其中某个协程,你可使用StopCoroutine.但在使用时,你需要注意一点,停止协程的方式要与开启协程的方式一致.StopCoroutine(“CustomCorutineFn”)必须与StartCoroutine(“CustomCorutineFn”)成对使用,与StartCoroutine(CustomCorutineFn())一起使用则完全无效.
通过StopCoroutine的重载方法可知道,还有两种方式可停止协程.在此举个例子,如下:

 IEnumerator cor;void Awake(){//注意保存IEnumerator变量.cor = Corutine_WaitForFixedUpdate();StartCoroutine(cor);StartCoroutine(Corutine_Stop());}IEnumerator Corutine_WaitForFixedUpdate(){Debug.Log(string .Format("A : {0}", 0));yield return new WaitForEndOfFrame();Debug.Log(string .Format("A : {0}", 1));}IEnumerator Corutine_Stop(){yield return new WaitForFixedUpdate();//通过cor停止协程//而不是this.StopCoroutine(Corutine_WaitForFixedUpdate());this.StopCoroutine(cor);}

如果想停止多个协程,可使用StopAllCoroutines方法,但这种方法只能停止当前MonoBehaviour类实例中所有协程.其他不受影响.
如果想停止gameObject上所有脚本组件中的协程,禁用脚本组件是无法停止协程的,只需要禁用gameObject即可.


如何用协程的输出10组计数(每组5次,每次1秒)?
源码如下:

using System.Collections;
using UnityEngine;public class mCorutine : MonoBehaviour
{public float time = 1;void Awake(){StartCoroutine(FirstLayerCorutine());}IEnumerator FirstLayerCorutine(){for (int i = 0; i < 10; i++){Debug.Log(string .Format("第{0}组", i + 1));yield return StartCoroutine(SecondLayerCorutine());}}IEnumerator SecondLayerCorutine(){for (int i = 0; i < 5; i++){Debug.Log(string .Format("{0}", i + 1));yield return new WaitForSeconds(time);}}
}

协程的使用还有很多要注意的地方,在这里分享一个关于协程运行时的监控和优化的链接.

本文由个人研究所得,如有错误,望大家指出~

本文标签: Unity3D