Unity 中的协同程序

今天咱就说说,协同程序coroutine。(这文章是在网吧敲的,没有unity,但是所有结论都被跑过,不管你信得过我还是信不过我,都要自己跑一下看看,同时欢迎纠错)
先说说啥是协程:协同程序是一个非常让人作呕的东西,它的表现形式非常像线程,对线程有过接触的朋友可能更理解我这句话的意思,你没接触过线程,那么理解它会有一些难度。但是它不存在线程安全问题,可以放心使用。这不是J哥信口雌黄空口白牙跟这猜的,事实是这样的:在操作系统层面,也就是更古老的大神们,觉得“并发”是一个很时髦的东西,很好使,于是他们允许操作系统中开启进程。后来,他们觉得一个进程中,如果也能并发该多好,那么线程应运而生(这些都是身为码农应该知道的一些常识:一个系统上可以运行多个进程,一个进程可以并发多个线程)。但是由于我们的unity是单线程的,那么还有一句不这么耳熟能详的话:一个线程可以多协程。于此看来,coroutine可以说是在unity主线程中“并发”的很多协程。这个“并发”为什么加引号,这将是我们研究的重要内容。

介绍了协同的定义,那么可以研究一下它在项目中究竟会怎么用。
下面就是面试考题中可能遇到的几种协同程序的用法:

1.不使用协同启动一个返回迭代器的函数。
void 家里来客人()

    沏茶();
     与客人聊天();

IEnumerator 沏茶();

2.在迭代器函数中,yield return 一个协同启动。 如

void 家里来客人()

     StartCoroutine(沏茶());
     与客人聊天();

IEnumerator 沏茶()
{
    yield return StartCoroutine(做水());
    找茶叶罐();
}
IEnumerator 做水();
void 找茶叶罐();

3.在迭代器函数中,直接启动一个协同。如
void 家里来客人()

     StartCoroutine(沏茶());
     与客人聊天();

IEnumerator 沏茶()
{
    StartCoroutine(做水());
    找茶叶罐();
}
IEnumerator 做水();
void 找茶叶罐();

这真的不是让大伙死记硬背或者是装逼,我写的很花哨,是为了让大伙动手敲一遍验证一下哈!!!!
下面逐条分析一下把。
对于第一条:这种用法,迭代器“沏茶”,根本不会被并发,比如在这种情况下,虽然沏茶是一个迭代器函数,但是你如果这么执行,也会是沏完了茶(彻底沏完了茶)再跟客人聊天。如果沏茶很费时间,那么不好意思,你在这段时间内,都不会跟客人聊天。这里迭代器等于白费。相当于普通函数。
对于第二条:你在startcoroutine之后,立即开始并发,也就是你一边执行沏茶,一边开始跟客人聊天了。现在进来看看你沏茶的时候。在沏茶的协同中,你先yield return 做水。这句话的意思就是,“等待做完水了”,再开始找茶叶罐。值得注意的是,你在做水,找茶叶罐的时候,已经开始跟客人聊天了。
对于第三条:你一遍执行沏茶,一遍与客人聊天。在你沏茶的时候,你是一边做水,一边找茶叶罐的。

对于这个例子来看呢。有这么几条收获:
1.yield return 跟return 没有任何关系,yield return xxx 翻译成人话就是“等待xxx返回之后”,是一个阻塞协同程序的操作。
2.开启协同程序,实现了伪并发,虽说看起来像并发,但是还是有先后执行次序,所以跟线程有本质的不同——线程理论上来说,先后次序是不可预知的——除非你用信号量等等进行人为控制哈。
3.开启协同程序,必须得是一个返回迭代的函数。否则编译不过。但是返回迭代的函数可以不在协同中调用,这样编译是通过的,但是基本不会这么写,代码不干净。

主要的骨头已经啃了,剩下一个就是协同程序什么时候才算完结?
1.碰见yield break——直接跳出携程,对某些判定失败必须跳出的时候,比如加载AssetBundle的时候,WWW都失败了,后边加载bundle没有必要了,这时候可以yield break。这个语句非常有用。 
2.执行到最后一行——最后一行不一定非得是 yield return xxx;我经常最后一句是一个 excute delegate什么的。
3*.补充:yield return null;yield return 0; 均不算完结协同程序!!!!

弄懂了这些,自己去揣摩揣摩,就能搞定协同啦。难点就是这些。它虽然恶心,但是并不难,希望对大家有帮助!

时间: 2024-10-04 16:26:33

Unity 中的协同程序的相关文章

【转】Unity中的协同程序-使用Promise进行封装(一)

原文:http://gad.qq.com/program/translateview/7170767 译者:陈敬凤(nunu)    审校:王磊(未来的未来) 每个Unity的开发者应该都对协同程序非常的熟悉.对于很多Unity的开发者而言,协同程序就是用来编写大量异步和延时任务的一种方法.如果你不在乎速度的话,有非常非常多的特殊方法可以在任何所需的时间暂停和恢复执行.在实践中,它们可以营造一种并发函数的幻觉 (虽然他们与线程无关!).然而,协同程序会有一些问题,许多程序员在使用协同程序的时候会

Lua中的协同程序

[前言] 协同程序与线程差不多,也就是一条执行序列,拥有自己独立的栈.局部变量和指令指针,同时又与其它协同程序共享全局变量和其它大部分东西.从概念上讲,线程与协同程序的主要区别在于,一个具有多个线程的程序可以同时运行几个线程,而协同程序却需要彼此协作的运行.就是说,一个具有多个协同程序的程序在任意时刻只能运行一个协同程序,并且正在运行的协同程序只会在其显式地要求挂起时,它的执行才会暂停. [协同程序基础] Lua将所有关于协同程序的函数放置在一个名为"coroutine"的table中

Lua中的协同程序 coroutine

Lua中的协程和多线程很相似,每一个协程有自己的堆栈,自己的局部变量,可以通过yield-resume实现在协程间的切换.不同之处是:Lua协程是非抢占式的多线程,必须手动在不同的协程间切换,且同一时刻只能有一个协程在运行.并且Lua中的协程无法在外部将其停止,而且有可能导致程序阻塞. 协同程序(Coroutine): 三个状态:suspended(挂起,协同刚创建完成时或者yield之后).running(运行).dead(函数走完后的状态,这时候不能再重新resume). coroutine

Lua中的协同程序 coroutine(转)

Lua中的协程和多线程很相似,每一个协程有自己的堆栈,自己的局部变量,可以通过yield-resume实现在协程间的切换.不同之处是:Lua协程是非抢占式的多线程,必须手动在不同的协程间切换,且同一时刻只能有一个协程在运行.并且Lua中的协程无法在外部将其停止,而且有可能导致程序阻塞. 协同程序(Coroutine): 三个状态:suspended(挂起,协同刚创建完成时或者yield之后).running(运行).dead(函数走完后的状态,这时候不能再重新resume). coroutine

Unity中自定义应用程序打开Assets目录下指定类型的文件

在Unity使用VS2017打开unityShader文件时总提示错误: 我也一直没找啥原因,我直接把之前工具拿过来了,shader文件直接使用vsCode打开,其他类型的文件也可这样处理,如:pdf,txt等文件均可 具体代码如下: 1 using System.Collections; 2 using System.Collections.Generic; 3 using UnityEditor; 4 using UnityEngine; 5 6 public class SetAssets

unity中开启和关闭协同程序

协同程序这个好处大大的,用过的人都说好 转载出处:http://wg2009perfect.blog.163.com/blog/static/127997663201211111222126/ 一.什么是协同程序 协同程序,即在主程序运行时同时开启另一段逻辑处理,来协同当前程序的执行.换句话说,开启协同程序就是开启一个线程. 二.协同程序的开启与终止 在Unity3D中,使用MonoBehaviour.StartCoroutine方法即可开启一个协同程序,也就是说该方法必须在MonoBehavi

转载 Unity3D协同程序(Coroutine)

摘要下: 1. coroutine, 中文翻译"协程".这个概念可能有点冷门,不过百度之,说是一种很古老的编程模型了,以前的操作系统里进程调度里用到过,现在操作系统的进程调度都是根据 时间片和优先级来进行轮换,以前是要程序自己来释放cpu的控制权,一直不释放一直也就占用着cpu,这种要求程序自己来进行调度的编程模型应该就叫"协 程"了. 协程和线程差不多,线程的调度是由操作系统完成的,协程把这项任务交给了程序员自己实现,当然也就可以提高灵活性,另外协程的开销比线程要

Lua语言基础汇总(7) -- 协同程序

前言 协同程序与线程差不多,也就是一条执行序列,拥有自己独立的栈.局部变量和指令指针,同时又与其它协同程序共享全局变量和其它大部分东西.从概念上讲,线程与协同程序的主要区别在于,一个具有多个线程的程序可以同时运行几个线程,而协同程序却需要彼此协作的运行.就是说,一个具有多个协同程序的程序在任意时刻只能运行一个协同程序,并且正在运行的协同程序只会在其显式地要求挂起时,它的执行才会暂停. 协同程序基础 Lua将所有关于协同程序的函数放置在一个名为“coroutine”的table中.函数create

《Lua程序设计》9.1 协同程序基础 学习笔记

协同程序(coroutine)与线程(thread)差不多,也就是一条执行序列,拥有自己独立的栈.局部变量和指令指针,同时又与其他协同程序共享全局变量和其他大部分东西.从概念上讲线程与协同程序的主要区别在于,一个具有多个线程的程序可以同时运行几个线程,而协同程序却需要彼此协作地运行.就是说,一个具有多个协同程序的程序在任意时刻只能运行一个协同程序,并且正在运行的协同程序只会在其显式地要求挂起(suspend)时,它的执行才会暂停. Lua将所有关于协同程序的函数放置在一个名为“coroutine