zoukankan      html  css  js  c++  java
  • Unity StartCoroutine 和 yield return 深入研究

    StartCoroutine和yield return表面意思很好理解,StartCoroutine就是开启一个协程,yield return 是迭代器块返回调用迭代的地方。

    是吧?不知道你什么感觉,反正我觉得,还是需要深入研究一下的。OK,here we go!

    首先,先看一下StartCoroutine在Unity官方的解释。

    意思是:一个协程的执行可以在任何地方用yield语句来暂停,yield return的值决定了什么时候协程恢复执行。协程在协调在几帧中执行的操作时有极大的用处.协程几乎没有任何性能开销。

    StartCoroutine一般都会立即返回,然而你也可以获得返回结果的值。但是这一步会等到协程结束执行才能生效。

    OK,意思应该不难理解,根据他的意思我们来分析一段程序。

    运行结果是:

    start1

    test1

    start2

    test2

    这下就一目了然了,当StartCoroutine刚调用的时候,可以理解为正常的函数调用,然后接着看调用的函数里面。

    当被调用函数执行到yield return null;(暂停协程,等待下一帧继续执行)时,根据Unity解释协同程序就会被暂停,其实我个人认为他这个解释不够精确,先返回开始协程的地方,然后再暂停协程。也就是先通知调用处,“你先走吧,不用管我”,然后再暂停协程。。怎么?不信?那我们再写个demo验证一下。

    执行结果:

    start1

    test1

    start2

    test2  (这个test2是等待三秒后才打印出来的)

    正好顺便验证了“yield return的值决定了什么时候协程恢复执行”这句,其实yield return后面的值可以后很多用法,可以看这个帖子:http://blog.sina.com.cn/s/blog_aaa4ce8d010131kr.html

    其实再回过头来想想,协程->协同程序->其实就是协同两个任务,表面看起来很简单,但是在一些稍微大点的项目中用起来,对于新手来说还是会有些晦涩。。。

    比如说

        IEnumerator Init()
        {
            yield return StartCoroutine(init1());
            Debug.Log("init1 finish");
            yield return StartCoroutine(init2());
            Debug.Log("init2 finish");
            yield return StartCoroutine(init3());
            Debug.Log("init3 finish");
        }
    
        IEnumerator init1()
        {
            // 模拟初始化
            yield return new WaitForSeconds(2);//
        }
        IEnumerator init2()
        {
            // do somthing..
            yield return new WaitForSeconds(2);//
        }
        IEnumerator init2()
        {
            // do somthing..
            yield return new WaitForSeconds(2);//
        }
    

    其实就是一个执行顺序的问题,这样调用能保证,init1,init2,init3一个一个的执行,不至于出现后面执行的代码引用一个前面未初始化的变量。。。

    那么,接着这个执行顺序的话题,我们在来研究一个话题,看下面代码

        void Start () {
            Debug.Log("start1");
            StartCoroutine(Test());
            Debug.Log("start2");
        }
    
        IEnumerator Test()
        {
            Debug.Log("test1");
            yield return StartCoroutine(DoSomething());
            Debug.Log("test2");
        }
    
        IEnumerator DoSomething()
        {
            Debug.Log("load 1");
            yield return null;
            Debug.Log("load 2");
        }

    执行结果:

    start1

    test1

    load1

    start2

    load2

    test2

    这种StartCoroutine中嵌套一个yield return StartCoroutine,第一个StartCoroutine会等到第二个StartCoroutine中所有代码结束后再继续执行,而第二个StartCoroutine中的yield语句会先返回第一个,然后立即返回他的调用处,也就是调用处会继续执行,而第一个StartCoroutine会等待第二个执行完再继续执行。

    如果还想继续深入,可以看一下C#中的迭代器,那里面说明了yield和IEnumerator根本到底是什么。

    OK,今天就说到这。Good Luck!

  • 相关阅读:
    python基础练习题(题目 学习使用auto定义变量的用法)
    python基础练习题(题目 模仿静态变量的用法)
    roaring bitmap 与 bitmap 比较. 编译运行
    Linux 实现开关机测试,记录开机次数
    Linux 显示开机欢迎信息(/etc/issue 与 /etc/motd)
    Linux 修改文件权限概述
    把编译好的程序放在Linux系统里,实现不需要配置程序和动态库的环境变量,直接执行
    Linux 提示 is not a symbolic link 错误解决方法
    Ubuntu 自动更新详解【转】
    C#线程入门
  • 原文地址:https://www.cnblogs.com/fly-100/p/3910515.html
Copyright © 2011-2022 走看看