协程,即Coroutine,顾名思义,协助程序的意思。我们在进行主任务的同时,需要一些分支任务来配合工作,这就是协程的用处。协程不是进程或线程,它是一个特殊的函数,可以认为它是一个返回值是IEnumerator(不知道也没关系,后面会说)的函数。协程依然是在主线程上进行的,是一种异步多任务处理的方式,相比于线程,开辟多个协程开销不大,适合对某任务进行分时处理。
我们只要知道协程是一个可以暂停执行,暂停后回到主函数,执行主函数剩余的部分,直到中断指令完成后,从中断指令的下一行继续执行协程剩余的函数就行。
首先我们要知道协程是通过迭代器实现的。什么是迭代器?迭代器是一种设计模式,可以让开发人员无需关心容器对象的底层架构,就可以遍访这个容器对象。简单来说,迭代器就是用来遍历一个序列中的所有对象。
在C#中可以使用foreach关键字就可以枚举一个序列
foreach (var item in collection)
{
Console.WriteLine(item?.ToString());
}
但foreach语句并非完美无缺,它依赖于.NET Core库中的两个接口:IEnumerable和IEnumerator
IEnumerable是可枚举的意思,IEnumerator是枚举器的意思
public interface IEnumerable
{
IEnumerator GetEnumerator();
}
继承这个接口需要实现暴露出来的GetEnumerator方法,返回一个IEnumerator对象
public interface IEnumerator
{
object Current { get; }
bool MoveNext();
void Reset();
}
IEnumerator接口有三个东西,current返回当前序列的元素,方法MoveNext()移动到下一个元素,Reset方法重置,所以继承这个接口需要实现这三个东西
从这个两个接口对比就可以发现,对于枚举一个容器,起真正作用是IEnumerator
所以一个对象只要实现IEnumerator接口就能遍历
下面来看一个实例
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace Csharp_study.Day1
{
//枚举对象
public class Anim
{
public string name;//动物的名字
//构造方法,对name赋值
public Anim(string name)
{
this.name = name;
}
}
//实现IEnumerator接口
public class MIEnumerator:IEnumerator
{
Anim[] anim;
int idx = -1;
//构造方法,对t赋值
public MIEnumerator(Anim[] t)
{
anim = t;
}
//实现IEnumerator接口的Current方法,获取当前元素的值
public object Current
{
get
{
if (idx == -1)
return new IndexOutOfRangeException();
return anim[idx];
}
}
//实现IEnumerator接口的MoveNext方法,向下一个元素移动
public bool MoveNext()
{
idx++;
return anim.Length > idx;
}
//实现IEnumerator接口的Reset方法,重置迭代器状态
public void Reset()
{
idx = -1;
}
}
class Class1
{
static void Main(string[] args)
{
//初始化一个Anim序列,用来遍历
Anim[] anims = new Anim[] { new Anim("老虎"),new Anim("大象"),new Anim("河马")};
MIEnumerator enumerator = new MIEnumerator(anims);
while(enumerator.MoveNext())
{
Anim test = enumerator.Current as Anim;
show(test);
}
void show(Anim p)
{
Console.WriteLine("这个小动物的名字是:" + p.name);
}
Console.ReadLine();
}
}
}
输出结果
这个小动物的名字是:老虎
这个小动物的名字是:大象
这个小动物的名字是:河马
从这个例子中就可以看出来,我们通过继承这个IEnumerator接口,然后实现它的Current,MoveNext和Reset方法就可以遍历这个Anim对象了。
所以不难看出,foreach关键字就是主要依靠IEnumerator接口实现,这个就不深入讲了,我们只要知道IEnumerator就行
此外在迭代器中还有一个关键字需要我们掌握-yield。yield是一个语法糖,是为了简化迭代器的实现语法才产生的,从上面的讲解不难发现,实际起作用的就是MoveNext和Current方法。所以C#2提供一个处理方法:yield语句。
这里就不细讲了,详细请看这里:C#迭代器的详细用法_真的没事鸭的博客-CSDN博客
IEnumrator 函数名(形参表) //最多只能有一个形参
{
yield return xxx; //恢复执行条件
//方法体
}
在IEnumerator类型的方法中写入需要执行的操作,遇到yield会暂时挂起,yield return后条件满足才继续执行后面的内容
yield return表示在迭代中下一个迭代时返回的数据,其中还有yield break表示跳出迭代
开启协程需要使用StartCoroutine()方法:
结束协程有两种情况:
终止协程有两种情况
比如我们在start函数定义了一个协程,首先第一帧我们在start函数开启协程,从上到下执行协程里面的操作,遇到yield return XXX,主函数是不受影响,主函数一直在执行。yield return后的条件满足后先挂起,在下一帧再继续执行后面的操作
下名看一个案例:实现一个秒表的效果,没过一秒数字增加1
我们在Hierarchy界面添加一个Text,注意这个是旧的Text,不是TextMeshPro
调整一下位置
然后建立一个C#脚本,下面编写一个这个脚本
using System.Collections;
using System.Collections.Generic;
using TMPro;
using UnityEngine;
using UnityEngine.UI;
public class test : MonoBehaviour
{
public Text text;
void Start()
{
StartCoroutine(Timer(1));//开启协程
}
IEnumerator Timer(float second)
{
int count = 0;
while (true)
{
yield return new WaitForSeconds(second);//等待一秒钟执行后续代码
count++;
Debug.Log("输出");
text.text = count.ToString();
}
}
}
然后我们将这个代码挂载Text所属的canvas,因为上面脚本获取的Text是pulbic,所以需要我们拖一个Text过去,所以把这个Text拖给canvas上的脚本
执行就可以发现,Text上的数字会自动加1了
如有错漏之处,敬请指正!