Unity C#笔记 协程详解(转)

摘要:
}Debug.Log;}但Unity封装了一个更好的类:WaitForSeconds使延迟协处理代码更加简洁使用WaitForSeconds的编写方法,yieldreurnnewWaitForseconds;该过程的使用示例如下所示。进程的使用示例如下:首先,在3秒后编写进程函数IEnumeratorTestWaitForSeconds(),执行Debug.Log;yieldreurnnewWaitForseconds;Debug.Log;},然后在某处使用StartCoroutine或StartCoroutine//启动进程:3秒后,执行Debug。logStartCoroutine//启动后,继续执行另一方面,Unity还有一个名为InvokeInvoke的函数,它也用于延迟调用。函数测试在延迟2秒后执行,但Invock调用的函数必须是null类型的返回值,并且必须是当前类中的方法。

目录

  • 什么是协程
  • 多线程
  • 协程
    • 协程的使用场景
    • 协程使用示例
    • Invoke的缺陷
  • 协程语法
    • 开启协程
    • 终止协程
    • 挂起
  • 协程的执行原理

什么是协程

在Unity中,协程(Coroutines)的形式是我最喜欢的功能之一,我都会使用它来控制需要定时的。

协同程序,在主程序运行的同时,开启另外一段逻辑处理,来协同当前程序的执行。
可能看了这段文字介绍还是有点模糊,其实可以用多线程来比较。

多线程

多线程,顾名思义,多条同时执行的线程。
最初,多线程的诞生是为了解决IO阻塞问题,如今多线程可以解决许多同样需要异步方法的问题(例如网络等)。
所谓异步,通俗点讲,就是我走我的线程,你走你的线程。当某个线程阻塞时,另一个线程不会受影响继续执行。

需要认识到的是,多线程并不是真正意义上的多条线程同时执行。
它的实际是将一个时间段分成若干个时间片,每个线程轮流运行一个时间片。

(如图,将执行步骤切分成极小的粒度,然后依次运行)

Unity C#笔记 协程详解(转)第1张

但是由于时间片粒度非常非常小,几乎看不出区别,所以程序执行效果跟真正意义上的并行执行效果基本一致。

多线程的缺陷

然而多线程有一个坏处,就是可能造成共享数据的冲突。

假如有一个变量i = 0, Step1_1的操作是进行++i操作,Step2_1的操作是进行--i操作。
我们预期最终结果i为0。

但由于操作切分得过小,可能会发生这样顺序的事:

  • 线程1:访问i, 将0存到寄存器
  • 线程2:访问i, 将0存到寄存器
  • 线程1:++i, 得到1
  • 线程2:--i, 得到-1
  • 线程1:将1写入到i的内存
  • 线程2:将-1写入到i的内存
  • 最终i的值为-1

当然多线程的冲突也有解决方案: 互斥锁....

但是这些多多少少会付出额外的代价,让程序变得臃肿。

协程

CPU有多条线程,一条线程可以有多个协程。

协程跟多线程类似,也有类似异步的效果(注意不是真正的异步)。
只不过它的切分粒度不是基于系统划分的时间片,而是基于我们编写的yield,而且往往粒度更大。

粒度是取决于自己定义什么时候让协程挂起:

//下面定义了一个协程函数,注意必须使用IEnumerator作为返还值才能成为协程函数。
IEnumerator Test()
{
  for(int i = 0; i<1000 ; ++i){
    ans += i;
    yield return 0;//挂起,下一帧再来从这个位置继续执行。
  }
  j+=2;
  yield return 0;//挂起,下一帧再来从这个位置继续执行。
  ++j;
  yield return 0;//挂起,下一帧再来从这个位置继续执行。
}

如果划分的粒度过大,协程所在的线程可能在相应的帧卡顿。
甚至如果让协程阻塞(死循环),那么协程所在的整个线程也会阻塞。
因此说协程可以有类似异步的效果,但是不是真正的异步。

Unity C#笔记 协程详解(转)第2张

协程的一大好处就是可以避免数据访问冲突的问题:
因为它的粒度相对多线程的大很多,所以往往很少出现冲突现象

在上面多线程的例子里,使用协程则可以这样:

  • Step1_1: 执行完++i, 此时i=1
  • Step2_1: 执行完--i, 此时i=0
  • 最终i的值为0

协程的使用场景

对于保证不会阻塞的并行操作且并行性要求不高的并行操作,可以使用协程。
更实际来说,协程最常用于延时执行等控制时间轴的操作,例如N秒后调用指定函数。

利用每帧执行一段协程的特性,我们可以引入个带累加计时判断循环,然后再超过3秒后跳出循环,执行Debug.Log()

//3s后执行Debug.Log
IEnumerator Test()
{
  for(float timer = 0.0f; timer < 3.0f ; timer += Time.DeltaTime){
    yield return 0;//挂起,下一帧再来从这个位置继续执行。
  }
  Debug.Log("启动协程3s后");
}

但是Unity封装了个更好用的类:WaitForSeconds
使这种延时的协程代码更加简洁。

  //原本写法
  for(float timer = 0.0f; timer < 3.0f ; timer += Time.DeltaTime){
    yield return 0;//挂起,下一帧再来从这个位置继续执行。
  }
  //使用WaitForSeconds的写法
  yield return new WaitForSeconds(3.0f);

协程使用示例

接下来就展示下,协程使用的示例:
首先编写好协程函数

IEnumerator TestWaitForSeconds()
{
    //3s后执行Debug.Log;
    yield return new WaitForSeconds(3.0f);
    Debug.Log("启动协程3s后");
}

然后在某个地方使用StartCoroutine(TestWaitForSeconds())或者StartCoroutine("TestWaitForSeconds")

  //启动协程:3s后执行Debug.log
  StartCoroutine(TestWaitForSeconds());
  //启动后,继续往下执行
  ...

Invoke的缺陷

另外一提,Unity还有个一样也是用于延时调用的函数,叫Invoke

Invoke("test",2.0f); \延时2秒后执行函数test

但是Invock所要调用的函数必须是空类型返还值,还必须得是在当前类里面的方法。

一般来说,用协程来解决这样的问题已经绰绰有余,而且还有更安全的调用方法而不是只用string类型作为参数的方法,因此没必要使用Invoke。

协程语法

开启协程

StartCoroutine(string methodName);
  • 参数是方法名(字符串类型),此方法可以包含一个参数。
  • 形参方法可以有返回值
StartCoroutine(IEnumerator method);
  • 参数是方法(TestMethod()),此方法中可以包含多个参数。
  • IEnumrator类型的方法不能含有ref或者out类型的参数,但可以含有被传递的引用
  • 形参方法必须有返回值,且返回值类型为IEnumrator,返回值使用(yield retuen +表达式或者值,或者 yield break)语句

终止协程

StopCoroutine(string methodName);//终止指定的协程
  • 在程序中调用StopCoroutine()方法只能终止以字符串形式启动的协程
StopAllCoroutine();//终止所有协程

挂起

//程序在下一帧中从当前位置继续执行
yield return 0;

//程序在下一帧中从当前位置继续执行
yield return null;

//程序等待N秒后从当前位置继续执行
yield return new WaitForSeconds(N);

//在所有的渲染以及GUI程序执行完成后从当前位置继续执行
yield new WaitForEndOfFrame();

//所有脚本中的FixedUpdate()函数都被执行后从当前位置继续执行
yield new WaitForFixedUpdate();

//等待一个网络请求完成后从当前位置继续执行
yield return WWW;

//等待一个xxx的协程执行完成后从当前位置继续执行
yield return StartCoroutine(xxx);

//如果使用yield break语句,将会导致协程的执行条件不被满足,不会从当前的位置继续执行程序,而是直接从当前位置跳出函数体,回到函数的根部
yield break;
协程的执行原理

协程函数的返回值时IEnumerator,它是一个迭代器,可以把它当成执行一个序列的某个节点的指针。
它提供了两个重要的接口,分别是Current(返回当前指向的元素)和MoveNext()(将指针向后移动一个单位,如果移动成功,则返回true)。

yield关键词用来声明序列中的下一个值或者是一个无意义的值。

如果使用yield return x(x是指一个具体的对象或者数值)的话,
那么MoveNext返回为true并且Current被赋值为x,如果使用yield break使得MoveNext()返回为false。
如果MoveNext函数返回为true意味着协程的执行条件被满足,则能够从当前的位置继续往下执行。否则不能从当前位置继续往下执行。

作者:KillerAery出处:http://www.cnblogs.com/KillerAery/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

免责声明:文章转载自《Unity C#笔记 协程详解(转)》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇cocos2dx 触摸测试一 单点和多点Linux命令整理-Kali下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Java之协程(quasar)

  一、前面我们简单的说了一下,Python中的协程原理。这里补充Java的协程实现过程。有需要可以查看python之协程。   二、Java协程,其实做Java这么久我也没有怎么听过Java协程的东西,但是一直有有听到微线程/协程的概念,这不在学习Python的时候接触到了协程一词。然后返回来去了解Java的协程问题,但是看了很多资料,发现官网以及很多地...

java多线程读取、操作List集合

import java.util.ArrayList; import java.util.List; import org.apache.commons.lang3.ArrayUtils;   public class Test_4 {     /**      * 多线程处理list      *       * @param ...

[转载]iOS 开发中为什么更新UI都要放在主线程中?

原因有2个: 1、在子线程中是不能进行UI 更新的,而可以更新的结果只是一个幻像:因为子线程代码执行完毕了,又自动进入到了主线程,执行了子线程中的UI更新的函数栈,这中间的时间非常的短,就让大家误以为分线程可以更新UI。如果子线程一直在运行,则子线程中的UI更新的函数栈 主线程无法获知,即无法更新 2、只有极少数的UI能,因为开辟线程时会获取当前环境,如点...

C# 多线程(lock,Monitor,Mutex,同步事件和等待句柄)

本篇从 Monitor,Mutex,ManualResetEvent,AutoResetEvent,WaitHandler 的类关系图开始,希望通过本篇的介绍能对常见的线程同步方法有一个整体的认识,而对每种方式的使用细节,适用场合不会过多解释。 让我们来看看这几个类的关系图: 1. lock 关键字 lock 是 C# 关键词,它将语句块标记为临界区,...

1-多线程与多进程

  一 进程与线程的概念 1.1 进程 考虑一个场景:浏览器,网易云音乐以及notepad++ 三个软件只能顺序执行是怎样一种场景呢?假如有两个程序A和B,程序A在执行到一半的过程中,需要读取大量的数据输入(I/O操作),而此时CPU只能静静地等待任务A读取完数据才能继续执行,这样就白白浪费了CPU资源。你是不是已经想到在程序A读取数据的过程中,让程序B去...

JUC 并发编程--04 常用的辅助类CountDownLatch , CyclicBarrier , Semaphore , 读写锁 , 阻塞队列,CompletableFuture(异步回调)

CountDownLatch 相当于一个减法计数器, 构造方法指定一个数字,比如6, 一个线程执行一次,这个数字减1, 当变为0 的时候, await()方法,才开始往下执行,, 看这个例子 CyclicBarrier 的用法, 字面意思:循环栅栏, 这是构造方法, 第一个参数parties 是线程数量, 第二个参数是barrierAction:...