Unity3D协程
协程介绍
Unity的协程系统是基于C#的一个简单而强大的接口 ,IEnumerator,它允许你为自己的集合类型编写枚举器。这一点你不必关注太多,我们直接进入一个简单的例子来看看协程到底能干什么。首先,我们来看一下这段简单的代码
Yield
在Countdown方法中其他的都很好理解,除了两个部分:
l IEnumerator 的返回值
l For循环中的yield return
为了能在连续的多帧中(在这个例子中,3秒钟等同于很多帧)调用该方法,Unity必须通过某种方式来存储这个方法的状态,这是通过IEnumerator 中使用yield return语句得到的返回值,当你“yield”一个方法时,你相当于说了,“现在停止这个方法,然后在下一帧中从这里重新开始!”。
注意:用0或者null来yield的意思是告诉协程等待下一帧,直到继续执行为止。当然,同样的你可以继续yield其他协程,我会在下一个教程中讲到这些。
一些例子
协程在刚开始接触的时候是非常难以理解的,无论是新手还是经验丰富的程序员我都见过他们对于协程语句一筹莫展的时候。因此我认为通过例子来理解它是最好的方法,这里有一些简单的协程例子:
多次输出“Hello”
记住,yield return是“停止执行方法,并且在下一帧从这里重新开始”
1 IEnumerator SayHelloFiveTimes() 2 { 3 Yield return 0; 4 Debug.Log("Hello"); 5 Yield return 0; 6 Debug.Log("Hello"); 7 Yield return 0; 8 Debug.Log("Hello"); 9 Yield return 0; 10 Debug.Log("Hello"); 11 Yield return 0; 12 Debug.Log("Hello"); 13 } 14 15 //This will do the exact same thing as the above function! 16 IEnumerator SayHello5Times() 17 { 18 for(inti = 0; i < 5; i++) 19 { 20 Debug.Log("Hello"); 21 Yield return 0; 22 } 23 }
每一帧输出“Hello”,无限循环。。。
通过在一个while循环中使用yield,你可以得到一个无限循环的协程,这几乎就跟一个Update()循环等同
1 IEnumerator SayHelloEveryFrame() 2 { 3 while(true) 4 { 5 //1. Say hello 6 Debug.Log("Hello"); 7 8 //2. Wait until next frame 9 Yield return 0; 10 11 }//3. This is a forever-loop, goto 1 12 }
计时
...不过跟Update()不一样的是,你可以在协程中做一些更有趣的事:
1 IEnumerator CountSeconds() 2 { 3 int seconds = 0; 4 5 while(true) 6 { 7 for(float timer = 0; timer < 1; timer += Time.deltaTime) 8 Yield return 0; 9 10 seconds++; 11 Debug.Log(seconds +" seconds have passed since the Coroutine started."); 12 } 13 }
这个方法突出了协程一个非常酷的地方:方法的状态被存储了,这使得方法中定义的这些变量都会保存它们的值,即使是在不同的帧中。还记得这个教程开始时那些烦人的计时器变量吗?通过协程,我们再也不需要担心它们了,只需要把变量直接放到方法里面!
开始和终止协程
1 StartCoroutine(Countdown());
如果我们想要终止所有的协程,可以通过StopAllCoroutines()方法来实现,它的所要做的就跟它的名字所表达的一样。注意,这只会终止在调用该方法的对象中(应该是指调用这个方法的类吧)开始的协程,对于其他的MonoBehavior类中运行的协程不起作用。
如果我们有以下这样两条协程语句:
1 StartCoroutine(FirstTimer()); 2 StartCoroutine(SecondTimer());
那我们怎么终止其中的一个协程呢?在这个例子里,这是不可能的,如果你想要终止某一个特定的协程,那么你必须得在开始协程的时候将它的方法名作为字符串,就像这样:
1 StartCoroutine("FirstTimer"); 2 StartCoroutine("SecondTimer"); 3 4 StopCoroutine("FirstTimer");
协程的参数
抽象化一个协程的第一个方法是给它传递参数,协程作为一个函数方法来说,它自然能够传递参数。这里有一个协程的例子,它在特定的地方输出了特定的信息。
1 Using UnityEngine; 2 Using System.Collections; 3 4 Public class TimerExample : MonoBehaviour 5 { 6 Void Start() 7 { 8 //Log "Hello!" 5 times with 1 second between each log 9 StartCoroutine(RepeatMessage(5, 1.0f,"Hello!")); 10 } 11 12 IEnumerator RepeatMessage(int count,float frequency,string message) 13 { 14 for(int i = 0; i < count; i++) 15 { 16 Debug.Log(message); 17 for(float timer = 0; timer < frequency; timer += Time.deltaTime) 18 Yield return 0; 19 20 } 21 } 22 }
嵌套的协程
在此之前,我们yield的时候总是用0(或者null),仅仅告诉程序在继续执行前等待下一帧。协程最强大的一个功能就是它们可以通过使用yield语句来相互嵌套。
1 IEnumerator Wait(float duration) 2 { 3 for(float timer = 0; timer < duration; timer += Time.deltaTime) 4 Yield return 0; 5 } 6 IEnumerator SaySomeThings() 7 { 8 Debug.Log("The routine has started"); 9 Yield return StartCoroutine(Wait(1.0f)); 10 Debug.Log("1 second has passed since the last message"); 11 Yield return StartCoroutine(Wait(2.5f)); 12 Debug.Log("2.5 seconds have passed since the last message"); 13 } 14 15 //Our wait function 16 IEnumerator Wait(float duration) 17 { 18 for(float timer = 0; timer < duration; timer += Time.deltaTime) 19 Yield return 0; 20 } 21 }
第二个方法用了yield,但它并没有用0或者null,而是用了Wait()来yield,这相当于是说,“不再继续执行本程序,直到Wait程序结束”。
控制对象行为的例子
在最后一个例子中,我们就来看看协程如何像创建方便的计时器一样来控制对象行为。协程不仅仅可以使用可计数的时间来yield,它还能很巧妙地利用任何条件。将它与嵌套结合使用,你会得到控制游戏对象状态的最强大工具。
运动到某一位置
对于下面这个简单脚本组件,我们可以在Inspector面板中给targetPosition和moveSpeed变量赋值,程序运行的时候,该对象就会在协程的作用下,以我们给定的速度运动到给定的位置。
1 usingUnityEngine; 2 Using System.Collections; 3 4 Public class MoveExample : MonoBehaviour 5 { 6 ublic Vector3 targetPosition; 7 ublic float moveSpeed; 8 9 Void Start() 10 { 11 StartCoroutine(MoveToPosition(targetPosition)); 12 } 13 14 IEnumerator MoveToPosition(Vector3 target) 15 { 16 while(transform.position != target) 17 { 18 transform.position = Vector3.MoveTowards(transform.position, target, moveSpeed * Time.deltaTime); 19 Yield return 0; 20 } 21 } 22 }
这样,这个程序并没有通过一个计时器或者无限循环,而是根据对象是否到达指定位置来yield。
按指定路径前进
我们可以让运动到某一位置的程序做更多,不仅仅是一个指定位置,我们还可以通过数组来给它赋值更多的位置,通过MoveToPosition() ,我们可以让它在这些点之间持续运动。
1 Using UnityEngine; 2 Using System.Collections; 3 4 Public class MoveExample : MonoBehaviour 5 { 6 ublic Vector3[] path; 7 ublic float moveSpeed; 8 9 Void Start() 10 { 11 StartCoroutine(MoveOnPath(true)); 12 } 13 14 IEnumerator MoveOnPath(bool loop) 15 { 16 do 17 { 18 foreach(var point in path) 19 Yield return StartCoroutine(MoveToPosition(point)); 20 } 21 while(loop); 22 } 23 24 IEnumerator MoveToPosition(Vector3 target) 25 { 26 while(transform.position != target) 27 { 28 transform.position = Vector3.MoveTowards(transform.position, target, moveSpeed * Time.deltaTime); 29 Yield return 0; 30 } 31 } 32 }
我还加了一个布尔变量,你可以控制在对象运动到最后一个点时是否要进行循环。
把Wait()程序加进来,这样就能让我们的对象在某个点就可以选择是否暂停下来,就像一个正在巡逻的AI守卫一样,这真是锦上添花啊!
注意:
如果你刚接触协程,我希望这两个教程能帮助你了解它们是如何工作的,以及如何来使用它们。以下是一些在使用协程时须谨记的其他注意事项:
- l 在程序中调用StopCoroutine()方法只能终止以字符串形式启动(开始)的协程;
- l 多个协程可以同时运行,它们会根据各自的启动顺序来更新;
- l 协程可以嵌套任意多层(在这个例子中我们只嵌套了一层);
- l 如果你想让多个脚本访问一个协程,那么你可以定义静态的协程;
- l 协程不是多线程(尽管它们看上去是这样的),它们运行在同一线程中,跟普通的脚本一样;
- l 如果你的程序需要进行大量的计算,那么可以考虑在一个随时间进行的协程中处理它们;
- l IEnumerator类型的方法不能带ref或者out型的参数,但可以带被传递的引用;
- l 目前在Unity中没有简便的方法来检测作用于对象的协程数量以及具体是哪些协程作用在对象上。