简介
什么是协程? ????????unity协程是一个能够暂停协程执行,暂停后立即返回主函数,执行主函数剩余的部分,直到中断指令完成后,从中断指令的下一行继续执行协程剩余的函数。函数体全部执行完成,协程结束。由于中断指令的出现,使得可以将一个函数分割到多个帧里去执行。
协程和线程的区别? ????????线程是利用多核达到真正的并行计算,缺点是会有大量的锁、切换、等待的问题,而协程是非抢占式,需要用户自己释放使用权来切换到其他协程, 因此同一时间其实只有一个协程拥有运行权, 相当于单线程的能力。
- 协程是 C# 线程的替代品, 是 Unity 不使用线程的解决方案。
- 使用协程不用考虑同步和锁的问题。
- 多个协程可以同时运行,它们会根据各自的启动顺序来更新。
性能: ????????在性能上相比于一般函数没有更多的开销。
协程的好处: ????????让原来要使用异步 + 回调方式写的非人类代码, 可以用看似同步的方式写出来。能够分步做一个比较耗时的事情,如果需要大量的计算,将计算放到一个随时间进行的协程来处理,能分散计算压力 协程的坏处: ????????协程本质是迭代器,且是基于unity生命周期的,大量开启协程会引起gc。如果同时激活的协程较多,就可能会出现多个高开销的协程挤在同一帧执行导致的卡帧。
使用
开启协程
// 形式一
StartCoroutine(CustomCorutine());
StartCoroutine(CustomCorutine("haha"));//向方法中传递参数
// 形式二
StartCoroutine(“CustomCorutine”);
StartCoroutine(“CustomCorutine”, "haha");//向方法中传递参数
void Awake()
{
StartCoroutine(Corutine_WaitForFixedUpdate());
}
void FixedUpdate()
{
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));
}
?
协程中的等待函数
yield null:协程将在下一帧所有脚本的Update执行之后,再继续执行.
yield WaitForSeconds:协程在延迟指定时间,且当前帧所有脚本的 Update全都执行结束后才继续执行.
yield WaitForFixedUpdate:协程在所有脚本的FixedUpdate执行之后,再继续执行.
yield WWW:协程在WWW下载资源完成后,再继续执行.
yield StartCoroutine:协程在指定协程执行结束后,再继续执行.
WaitForSecondsRealtime:与WaitForSeconds类似,但不受时间缩放影响.
WaitWhile:当返回条件为假时才执行后续步骤.
停止协程
????????在开发中可能会开启多个协程,如果你想停止其中某个协程,你可使用StopCoroutine,但在使用时,停止协程的方式要与开启协程的方法一致。.StopCoroutine(“CustomCorutine”)必须与StartCoroutine(“CustomCorutine”)成对使用,与StartCoroutine(CustomCorutine())一起使用则完全无效。
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即可。
其他问题
协程书写时的性能优化: ????????常见的问题是直接new 一个中断指令,带来不必要的 GC 负担,可以复用一个全局的中断指令对象,优化掉开销;在 Yielders.cs 这个文件里,已经集中地创建了上面这些类型的静态对象 这个链接分析了一下https://blog.csdn.net/liujunjie612/article/details/70623943
协程是在什么地方执行? ????????协程不是线程,不是异步执行;协程和monobehaviour的update函数一样也是在主线程中执行。unity在每一帧都会处理对象上的协程,也就是说,协程跟update一样都是unity每帧会去处理的函数。经过测试,协程至少是每帧的lateUpdate后运行的。

????????详细信息请看官方文档:Unity - Manual: Order of execution for event functions
同一时刻同一脚本实例中能有多少个运行的协程? ????????在一个MonoBehaviour提供的主线程里只能有一个处于运行状态的协程。因为协程不是线程,不是并行的。同一时刻、一个脚本实例中可以有多个暂停的协程,但只有一个运行着的协程。
协程的注意点:
- IEnumerator 类型的方法不能带 ref 或者 out 型的参数,但可以带被传递的引用。
- 在函数 Update 和 FixedUpdate 中不能使用 yield 语句,否则会报错, 但是可以启动协程。
- 在一个协程中,StartCoroutine()和 yield return StartCoroutine()是不一样的。
????????前者仅仅是开始一个新的Coroutine,这个新的Coroutine和现有Coroutine并行执行。 后者是返回一个新的Coroutine,是一个中断指令,当这个新的Coroutine执行完毕后,才继承执行现有Coroutine。
参考博客:https://zhuanlan.zhihu.com/p/59619632
????????????????? https://www.cnblogs.com/crazytomato/p/8178382.html
|