Unity自己实现协程调度

自己实现协程调度有几个好处:

  1. 脱离Unity独立,拿到别的地方也可以用。
  2. 非主线程也可以启动协程,然后在主线程执行,比如异步网络消息等。
  3. 可以给每个协程一个id,通过id随时启动或关闭某个特定的协程,或者非MonoBehavior对象也可以管理属于自己的协程。

 

Unity中,Coroutine是在LateUpdate执行的,每一个update都会执行一部分代码,拿IEnumerator来说,就是每一次都会MoveNext一下。

IEnumerator有三个接口:

  • Current:返回一个object,可以设置当前的一个状态。
  • MoveNext:返回true表示没有到最后,返回false表示已经完成枚举。
  • Reset:恢复状态,从头开始枚举。

 

IEnumerator方法内部会有一个状态机的实现 ,如果StartCoroutine一个返回IEnumerator方法,每次就会MoveNext到一个yield,遇到yield return,MoveNext会返回true,Current值就是return的对象(yield return null时Current=null, yield return obj时Current=obj),遇到yield break,MoveNext会返回false,表示已经执行完毕。但是如果直接调用IEnumerator.MoveNext,不会去对里面的另一个协程MoveNext,下次就跳过了。

 

基于以上内容,可以自己写一个方法对IEnumerator进行MoveNext,以实现自制协程的核心代码:

public static bool MoveNext(IEnumerator subTask)
{
  var child = subTask.Current;
  //yield return另一个协程:递归MoveNext
  if (child != null && child is IEnumerator && MoveNext(child as IEnumerator))
    return true;
  #if UNITY
  //yield return www:等待www完成
  if(child is UnityEngine.WWW && !(child as UnityEngine.WWW).isDone)
    return true;
  #endif
  if (subTask.MoveNext ())
    return true;
   return false;
}

 

管理协程很简单,用一个链表来管理:

LinkedListNode<IEnumerator> node = m_taskList.First;
LinkedListNode<IEnumerator> tempNode = node;
while (node != null) {                
  if (!MoveNext(node.Value)) {
    tempNode
= node; node = node.Next; //此处可以写删除节点的后续处理 m_taskList.Remove(tempNode); } else { node = node.Next; }
}

 

自己写IEnumerator方法时,return另一个协程不用写yield return StartCoroutine(func()),直接写yield return func()就可以,如:

IEnumerator a()
{
  ...
}

IEnumerator b()
{
  ...
  yield return a();
  ...
}

 

也可以自己写一些实用的IEnumerator类,比如我们项目里用的WaitForEvent, 还有组合类如WaitForAll,WaitForAny等。

posted @ 2017-02-05 18:29  DrAsh9N  阅读(2899)  评论(0编辑  收藏  举报