什么是协程

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

协同程序,在主程序运行的同时,开启另外一段逻辑处理,来协同当前程序的执行。

可能看了这段文字介绍还是有点模糊,其实可以用多线程来比较。

多线程

多线程,顾名思义,多条同时执行的线程。

最初,多线程的诞生是为了解决IO阻塞问题,如今多线程可以解决许多同样需要异步方法的问题(例如网络等)。

所谓异步,通俗点讲,就是我走我的线程,你走你的线程。当某个线程阻塞时,另一个线程不会受影响继续执行。

需要认识到的是,多线程并不是真正意义上的多条线程同时执行。

它的实际是将一个时间段分成若干个时间片,每个线程轮流运行一个时间片。

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

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

多线程的缺陷

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

假如有一个变量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;//挂起,下一帧再来从这个位置继续执行。
}

如果划分的粒度过大,协程所在的线程可能在相应的帧卡顿。

甚至如果让协程阻塞(死循环),那么协程所在的整个线程也会阻塞。

因此说协程可以有类似异步的效果,但是不是真正的异步。

协程的一大好处就是可以避免数据访问冲突的问题:

因为它的粒度相对多线程的大很多,所以往往很少出现冲突现象

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

  • 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;

协程的执行原理

TODO

Unity C#笔记 协程的更多相关文章

  1. Unity中的协程(一)

    这篇文章很不错的问题,推荐阅读英文原版: Introduction to Coroutines Scripting with Coroutines   这篇文章转自:http://blog.csdn. ...

  2. Unity中的协程是什么?

    什么是协程? 1.协程是一个分部执行,遇到条件(yield return 语句)会挂起,直到条件满足才会被唤醒继续执行后面的代码. 2.Unity在每一帧(Frame)都会去处理对象上的协程.Unit ...

  3. Unity中使用协程进行服务端数据验证手段

    近期在做项目中的个人中心的一些事情,用户头像上传,下载,本地缓存,二级缓存,压缩,这些都要做,麻雀虽小五脏俱全啊,也是写的浑浑噩噩的, 当我们在上传用户头像的时候,向服务端发送上传头像请求之前,一般都 ...

  4. 用C# 模拟实现unity里的协程

    注:需要了解C#的迭代器,不然很难理解. 之前面试有被问到unity协程的原理,以及撇开unity用纯C#去实现协程的方法.后来了解一下,确实可以的.趁这会有空,稍微总结一下. 还是结合代码说事吧: ...

  5. unity 自实现协程总结

    unity本人自实现了一个协程调用. 只是moveNext()的简单协程调用和封装,这个没什么好说的, 网上例子一大堆. 但使用的过程中遇到了几个问题. 1. 自己写的moveNext() 协程不能等 ...

  6. 学习python笔记 协程

    下面将一个经典的消费者和生产者的案例进行分析: import time def consumer(): r = '' while True: n = yield r if not n: return ...

  7. Unity XLua之协程

    如何使用xlua实现协程,示例代码如下: 转载请注明出处:https://www.cnblogs.com/jietian331/p/10735773.html local unpack = unpac ...

  8. xlua 实现协程替换Unity中的协程

    C#中的协程: IEnumerator ShowSpiritInfo() { UIMessageMgr.ShowMsgWait(true); DestroyUIModelInfo(); bool is ...

  9. python学习笔记 协程

    在学习异步IO模型前,先来了解协程 协程又叫做微线程,Coroutine 子程序或者成为函数,在所有语言中都是层级调用,比如a调用b,b调用c.c执行完毕返回,b执行完毕返回,最后a执行完毕返回 所以 ...

随机推荐

  1. 简明4步,让Python的好朋友Pycharm变得更加雅观!

    跟着人工智能/机械学习的兴起,Python再次取得广泛程序员的关注.而JetBrains出品的PyCharm无疑是最好用的Python IDE之一. 把本身的IDE装备安排得既有逼格又雅观,这是每个P ...

  2. Java并发之AQS详解

    一.概述 谈到并发,不得不谈ReentrantLock:而谈到ReentrantLock,不得不谈AbstractQueuedSynchronizer(AQS)! 类如其名,抽象的队列式的同步器,AQ ...

  3. 团队项目第二阶段个人进展——Day6

    一.昨天工作总结 冲刺第六天,学习了leancloud的一些数据处理知识,并看了如何在微信小程序中使用 二.遇到的问题 无 三.今日工作规划 通过动手完成一个demo来学习后端数据的请求和响应

  4. Asp.Net MVC 中JS通过ajaxfileupload上传图片获取身份证姓名、生日、家庭住址等详细信息

    客户要求用身份证图片上传获取身份证的详细信息就下来研究了一下(现在的客户真的懒 身份证信息都懒得输入了哈哈...),经过慢慢研究,果然皇天不负有心人搞出来了.这个借助的是腾讯的一个SKD  腾讯优图云 ...

  5. maven安装和配置及创建maven项目

    (1)下载maven,下载成功后,解压到本地磁盘 里面包含这几项 (2)配置maven环境变量MAVEN_HOME.path (3)最后检验配置是否成功:用win键+R,来打开命令行提示符窗口,即Do ...

  6. capwap学习笔记——capwap的前世今生(转)

    1 capwap的前世今生 1.1 胖AP.瘦AP.AC 传统的WLAN网络都是为企业或家庭内少量移动用户的接入而组建的.因此,只需要一个无线路由器就可以搞定了,就好像现在家用的无线路由器就是胖AP. ...

  7. mysql管理工具navicat的快捷键

    1. ctrl + q  或者 ctrl+n: 打开新查询窗口 2. ctrl + r: 运行当前窗口内的所有语句 3. ctrl + shit + r: 只运行选中的语句 4. ctrl + w: ...

  8. JavaScript-点击任意点显示隐藏

    //开/关 var only = document.getElementById('only'); var centerBox = document.getElementById('centerBox ...

  9. 微机原理基础(四)—— MSC51

    一.MCS51基本组成(STC89C52) CPU(8051CPU) + 存储器(4KB ROM/256B RAM)+外设(4组IO口,两个定时器,一个串口) 1.组成结构简图             ...

  10. 6.app架构基础

    app架构,一个听起来高大尚的名字,很多小伙伴听到这个词语感觉很迷茫,不知道架构具体说的是啥?在q群里,"app后端应该怎么架构"这个问题被问了无数次.通过阅读本文,根据本人提出的 ...