【吐血推荐】简要分析unity3d中剪不断理还乱的yield

  在学习unity3d的时候很容易看到下面这个例子:


1 void Start () {
2 StartCoroutine(Destroy());
3 }
4
5 IEnumerator Destroy(){
6 yield return WaitForSeconds(3.0f);
7 Destroy(gameObject);
8 }

  这个函数干的事情很简单:调用StartCoroutine函数开启协程,yield等待一段时间后,销毁这个对象;由于是协程在等待,所以不影响主线程操作。一般来说,看到这里的时候都还不会晕,yield就是延时一段时间以后继续往下执行呗,恩,学会了,看着还蛮好用的。

====================================================分割线====================================================

  当然,yield能干的事情远远不止这种简单的特定时间的延时,例如可以在下一帧继续执行这段代码(yield
return null),可以在下一次执行FixedUpdate的时候继续执行这段代码(yield new WaitForFixedUpdate
();),可以让异步操作(如LoadLevelAsync)在完成以后继续执行,可以……可以让你看到头晕。

  unity3d官方对于协程的解释是:一个协同程序在执行过程中,可以在任意位置使用yield语句。yield的返回值控制何时恢复协同程序向下执行。协同程序在对象自有帧执行过程中堪称优秀。协同程序在性能上没有更多的开销。StartCoroutine函数是立刻返回的,但是yield可以延迟结果。直到协同程序执行完毕。(原文:The
execution of a coroutine can be paused at any point using the yield statement.
The yield return value specifies when the coroutine is resumed. Coroutines are
excellent when modelling behaviour over several frames. Coroutines have
virtually no performance overhead. StartCoroutine function always returns
immediately, however you can yield the result. This will wait until the
coroutine has finished execution.)

  如果只是认为yield用于延时,那么可以用的很顺畅;但是若看到yield还有这么多功能,目测瞬间就凌乱了,更不要说活学活用了。不过,如果从原理上进行理解,就很容易理清yield的各种功能了。

C#中的yield


 1 public static IEnumerable<int> GenerateFibonacci()
2 {
3 yield return 0;
4 yield return 1;
5
6 int last0 = 0, last1 = 1, current;
7
8 while (true)
9 {
10 current = last0 + last1;
11 yield return current;
12
13 last0 = last1;
14 last1 = current;
15 }
16 }

  yield return的作用是在执行到这行代码之后,将控制权立即交还给外部。yield
return之后的代码会在外部代码再次调用MoveNext时才会执行,直到下一个yield
return——或是迭代结束。虽然上面的代码看似有个死循环,但事实上在循环内部我们始终会把控制权交还给外部,这就由外部来决定何时中止这次迭代。有了yield之后,我们便可以利用“死循环”,我们可以写出含义明确的“无限的”斐波那契数列。转自老赵的博客

  IEnumerable与IEnumerator的区别比较小,在unity3d中只用到IEnumerator,功能和IEnumerable类似。至于他们的区别是什么,网上搜了半天,还是模糊不清,有童鞋能解释清楚的请留言。不过对于这段代码对于unity3d中yield的理解已经足够了。

游戏中需要使用yield的场景

  既然要使用yield,就得给个理由吧,不能为了使用yield而使用yield。那么先来看看游戏中可以用得到yield的场景:

  • 游戏结算分数时,分数从0逐渐上涨,而不是直接显示最终分数

  • 人物对话时,文字一个一个很快的出现,而不是一下突然出现

  • 10、9、8……0的倒计时

  • 某些游戏(如拳皇)掉血时血条UI逐渐减少,而不是突然降低到当前血量

…………………………

unity3d中yield应用举例

  首先是官网的一段代码:


 1 using UnityEngine;
2 using System.Collections;
3
4 public class yield1 : MonoBehaviour {
5
6 IEnumerator Do() {
7 print("Do now");
8 yield return new WaitForSeconds(2);
9 print("Do 2 seconds later");
10 }
11 void Awake() {
12 StartCoroutine(Do());
13 print("This is printed immediately");
14 }
15
16 // Use this for initialization
17 void Start () {
18
19 }
20
21 // Update is called once per frame
22 void Update () {
23
24 }
25 }

  这个例子将执行Do,但是Do函数之后的print指令会立刻执行。这个例子没有什么实际意义,只是为了验证一下yield确实是有延时的。

  下面来看看两段显示人物对话的代码(对话随便复制了一段内容),功能是一样的,但是方法不一样:


 1 using UnityEngine;
2 using System.Collections;
3
4 public class dialog_easy : MonoBehaviour {
5 public string dialogStr = "yield return的作用是在执行到这行代码之后,将控制权立即交还给外部。yield return之后的代码会在外部代码再次调用MoveNext时才会执行,直到下一个yield return——或是迭代结束。虽然上面的代码看似有个死循环,但事实上在循环内部我们始终会把控制权交还给外部,这就由外部来决定何时中止这次迭代。有了yield之后,我们便可以利用“死循环”,我们可以写出含义明确的“无限的”斐波那契数列。";
6 public float speed = 5.0f;
7
8 private float timeSum = 0.0f;
9 private bool isShowing = false;
10 // Use this for initialization
11 void Start () {
12 ShowDialog();
13 }
14
15 // Update is called once per frame
16 void Update () {
17 if(isShowing){
18 timeSum += speed * Time.deltaTime;
19 guiText.text = dialogStr.Substring(0, System.Convert.ToInt32(timeSum));
20
21 if(guiText.text.Length == dialogStr.Length)
22 isShowing = false;
23 }
24 }
25
26 void ShowDialog(){
27 isShowing = true;
28 timeSum = 0.0f;
29 }
30 }

  这段代码实现了在GUIText中逐渐显示一个字符串的功能,速度为每秒5个字,这也是新手常用的方式。如果只是简单的在GUIText中显示一段文字,ShowDialog()函数可以做的很好;但是如果要让字一个一个蹦出来,就需要借助游戏的循环了,最简单的方式就是在Update()中更新GUIText。

  从功能角度看,这段代码完全没有问题;但是从代码封装性的角度来看,这是一段很恶心的代码,因为本应由ShowDialog()完成的功能放到了Update()中,并且在类中还有两个private变量为这个功能服务。如果将来要修改或者删除这个功能,需要在ShowDialog()和Update()中修改,并且还可能修改那两个private变量。现在代码比较简单,感觉还不算太坏,一旦Update()中再来两个类似的的功能,估计写完代码一段时间之后自己修改都费劲。

  如果通过yield return null实现帧与帧之间的同步,则代码优雅了很多:


 1 using UnityEngine;
2 using System.Collections;
3
4 public class dialog_yield : MonoBehaviour {
5 public string dialogStr = "yield return的作用是在执行到这行代码之后,将控制权立即交还给外部。yield return之后的代码会在外部代码再次调用MoveNext时才会执行,直到下一个yield return——或是迭代结束。虽然上面的代码看似有个死循环,但事实上在循环内部我们始终会把控制权交还给外部,这就由外部来决定何时中止这次迭代。有了yield之后,我们便可以利用“死循环”,我们可以写出含义明确的“无限的”斐波那契数列。";
6 public float speed = 5.0f;
7
8 // Use this for initialization
9 void Start () {
10 StartCoroutine(ShowDialog());
11 }
12
13 // Update is called once per frame
14 void Update () {
15 }
16
17 IEnumerator ShowDialog(){
18 float timeSum = 0.0f;
19 while(guiText.text.Length < dialogStr.Length){
20 timeSum += speed * Time.deltaTime;
21 guiText.text = dialogStr.Substring(0, System.Convert.ToInt32(timeSum));
22 yield return null;
23 }
24 }
25 }

  相关代码都被封装到了ShowDialog()中,这么一来,不论是要增加、修改或删除功能,都变得容易了很多。

  根据官网手册的描述,yield return null可以让这段代码在下一帧继续执行。在ShowDialog()中,每次更新文字以后yield
return null,直到这段文字被完整显示。看到这里,可能有童鞋不解:

  • 为什么在协程中也可以用Time.deltaTime?

  • 协程中的Time.deltaTime和Update()中的一样吗?

  • 这样使用协程,会不会出现与主线程访问共享资源冲突的问题?(线程的同步与互斥问题)

  • yield return null太神奇了,为什么会在下一帧继续执行这个函数?

  • 这段代码是不是相当于为ShowDialog()构造了一个自己的Update()?

  要解释这些问题,先看看unity3d中的协程是怎么运行的吧。

协程原理分析

  本段内容转自这篇博客,想看的童鞋自己点击。

  首先,请你牢记:协程不是线程,也不是异步执行的。协程和 MonoBehaviour 的
Update函数一样也是在MainThread中执行的。使用协程你不用考虑同步和锁的问题。

  UnityGems.com给出了协程的定义:

  A coroutine is a function that is executed partially and,
presuming suitable conditions are met, will be resumed at some point in the
future until its work is done.

  协程是一个分部执行,遇到条件(yield return
语句)会挂起,直到条件满足才会被唤醒继续执行后面的代码。Unity在每一帧(Frame)都会去处理对象上的协程。Unity主要是在Update后去处理协程(检查协程的条件是否满足):

  从上图的剖析就明白,协程跟Update()其实一样的,都是Unity每帧对会去处理的函数(如果有的话)。如果MonoBehaviour
是处于激活(active)状态的而且yield的条件满足,就会协程方法的后面代码。还可以发现:如果在一个对象的前期调用协程,协程会立即运行到第一个 yield
return 语句处,如果是 yield return null
,就会在同一帧再次被唤醒。如果没有考虑这个细节就会出现一些奇怪的问题。
  注:图和结论都是从UnityGems.com
上得来的,经过验证发现与实际不符,D.S.Qiu用的是Unity 4.3.4f1
进行测试的。经过测试验证,协程至少是每帧的LateUpdate()后去运行

  协程其实就是一个IEnumerator(迭代器),IEnumerator 接口有两个方法 Current 和 MoveNext() ,迭代器方法运行到
yield return 语句时,会返回一个expression表达式并保留当前在代码中的位置。
当下次调用迭代器函数时执行从该位置重新启动。unity3d在每帧做的工作就是:调用协程(迭代器)MoveNext() 方法,如果返回 true
,就从当前位置继续往下执行。详情见这篇博客

  如果理解了这张图,之前显示人物对话的功能最后提到的那些疑惑也就很容易理解了:

  • 协程和Update()一样更新,自然可以使用Time.deltaTime了,而且这个Time.deltaTime和在Update()当中使用是一样的效果(使用yield
    return null的情况下)

  • 协程并不是多线程,它和Update()一样是在主线程中执行的,所以不需要处理线程的同步与互斥问题

  • yield return null其实没什么神奇的,只是unity3d封装以后,这个协程在下一帧就被自动调用了

  • 可以理解为ShowDialog()构造了一个自己的Update(),因为yield return
    null让这个函数每帧都被调用了

时间: 2024-12-15 01:38:16

【吐血推荐】简要分析unity3d中剪不断理还乱的yield的相关文章

【转】简要分析unity3d中剪不断理还乱的yield

在学习unity3d的时候很容易看到下面这个例子: 1 void Start () { 2 StartCoroutine(Destroy()); 3 } 4 5 IEnumerator Destroy(){ 6 yield return WaitForSeconds(3.0f); 7 Destroy(gameObject); 8 } 这个函数干的事情很简单:调用StartCoroutine函数开启协程,yield等待一段时间后,销毁这个对象:由于是协程在等待,所以不影响主线程操作.一般来说,看到

简要分析unity3d中剪不断理还乱的yield

在学习unity3d的时候非常easy看到以下这个样例: 1 void Start () { 2 StartCoroutine(Destroy()); 3 } 4 5 IEnumerator Destroy(){ 6 yield return WaitForSeconds(3.0f); 7 Destroy(gameObject); 8 } 这个函数干的事情非常easy:调用StartCoroutine函数开启协程.yield等待一段时间后,销毁这个对象:因为是协程在等待.所以不影响主线程操作.

Unity 3D中不得不说的yield协程与消息传递

1. 协程 在Unity 3D中,我们刚开始写脚本的时候肯定会遇到类似下面这样的需求:每隔3秒发射一个烟花.怪物死亡后20秒再复活之类的.刚开始的时候喜欢把这些东西都塞到Update里面去,就像下面这样写. 1 float nowTime = 3.0f; 2 bool isDead = true; 3 float deadTime = 20.0f; 4 5 void startFireworks() 6 { 7 // 放烟花 8 } 9 10 void revival() 11 { 12 //

Unity3D中常用的数据结构总结与分析

Unity3D中常用的数据结构总结与分析 c#语言规范 阅读目录 1.几种常见的数据结构 2.几种常见数据结构的使用情景 来到周末,小匹夫终于有精力和时间来更新下博客了.前段时间小匹夫读过一份代码,对其中各种数据结构灵活的使用赞不绝口,同时也大大激发了小匹夫对各种数据结构进行梳理和总结的欲望.正好最近也拜读了若干大神的文章,觉得总结下常用的数据结构以供自己也能灵活的使用变得刻不容缓.那么还是从小匹夫的工作内容入手,就谈谈在平时使用U3D时经常用到的数据结构和各种数据结构的应用场景吧. 回到目录

Unity3d中CharacterController的移动和碰撞分析

在Unity3d中系统提供的第一人称视角模型First Person Controller的移动可分为两种: 一.移动transform 这种移动方式为直接对该人物模型的transform属性做位移操作,移动方式为在Update函数中的写法: void Update () { if(Input.GetKey(KeyCode.W)) { transform.Translate(Vector3.forward * Time.deltaTime * speed); } else if(Input.Ge

机器学习资料与攻略超强整理吐血推荐(二)

在前文<机器学习资料与攻略超强整理吐血推荐(一)>中,我们讲到要进入人工智能的圈子,数学是不可逾越的第一关,然后我们又推荐了一些你必须掌握的数学方面的资料.接下来,在这一篇中,我们将介绍关于机器学习的理论与工具方面的资料. 二.工具篇 工欲善其事,必先利其器.大数据时代,人工智能的工具趋势是显而易见且确定无疑的.作为一名数据科学家,其实你可以选择的工具非常多,其实随着时代的发展和技术的演进,传统的SPSS.STATA和SAS等这些老牌数据分析工具都已经集成了机器学习和数据挖掘的模块.当然,这不

浅析游戏引擎的资源管理机制——扒一扒Unity3D中隐藏在背后的资源管理

      游戏中通常有大量资源,如网格.材质.纹理.动画.着色器程序和音乐等,游戏引擎作为做游戏的工具,自然要提供良好的资源管理,让游戏开发者用最简单的方式使用资源.游戏引擎的资源管理包括两大部分:离线资源管理和运行时资源管理.本文仅对前者进行简要介绍,并结合Unity3D和OGRE进行分析. 资源创作与导出 游戏中的资源由各种数字内容创作工具(DCC, digital content creation)进行创作,如: 三维模型:3ds Max,Maya等: 纹理:Photoshop等: 音乐

loosejar原理简要分析

loosejar这个小工具可以动态分析出应用中有每个jar包的实际使用情况,详情请参阅<通过loosejar清理应用中冗余的jar包>基本原理是利用instrumentation的特性用 Instrumentation,开发者可以构建一个独立于应用程序的代理程序(Agent),用来监测和协助运行在 JVM 上的程序,甚至能够替换和修改某些类的定义.有了这样的功能,开发者就可以实现更为灵活的运行时虚拟机监控和 Java 类操作了.关于instrumentation的详细介绍,可以参阅这篇文章&l

intel万兆网卡驱动简要分析

原创文章,转载请注明: 转载自pagefault 本文链接地址: intel万兆网卡驱动简要分析 这里分析的驱动代码是给予linux kernel 3.4.4 对应的文件在drivers/net/ethernet/intel 目录下,这个分析不涉及到很细节的地方,主要目的是理解下数据在协议栈和驱动之间是如何交互的. 首先我们知道网卡都是pci设备,因此这里每个网卡驱动其实就是一个pci驱动.并且intel这里是把好几个万兆网卡(82599/82598/x540)的驱动做在一起的. 首先我们来看对