浅谈Unity协程的工作机制

一. 什么是协程

协程概述

        在 Unity 中,协程(Coroutine)是一种非常常用的机制,用于非阻塞地处理需要跨越多个帧、等待某些条件或延迟一段时间才能完成的任务。Unity 的协程通过 C# 的 IEnumerator 和 yield return 实现,使得你可以在游戏主线程中以一种简洁的方式执行异步操作,而不需要使用复杂的多线程或回调。

        协程允许你暂停代码的执行,并在稍后的某个时间点恢复执行。适合处理游一些常见需求,比如:

延迟一定时间后执行代码。

等待某个条件(如动画完成、玩家输入)后继续执行。

逐帧执行任务,而不是一次性完成,避免阻塞主线程。

下面一些示例

void Start()
{
    // 启动协程
    StartCoroutine(MyCoroutine());
    Debug.Log("因为是协程,所以不会阻塞打印");
}

IEnumerator MyCoroutine()
{
    Debug.Log("Before waiting");

    // 暂停协程 2 秒
    yield return new WaitForSeconds(2);

    Debug.Log("After waiting 2 seconds");
}

执行流程:

        当 StartCoroutine(MyCoroutine()) 被调用时,MyCoroutine 方法开始执行,先输出 "Before waiting".

        在遇到yield return时会立即返回控制权给上级调用者,这里上级调用者是主线程,打印"因为是协程,所以不会阻塞打印",并且执行到这里的状态会被保存下来,以便条件(这里的条件是等待两秒)完成后继续向下执行.

        2 秒后,协程恢复,继续执行,输出 "After waiting 2 seconds"。

yield return 的使用

协程中的 yield return 是关键,它决定了协程什么时候暂停以及在什么条件下恢复。

yield return null://暂停协程直到下一帧。它相当于让协程在下一帧继续执行。

yield return new WaitForSeconds(x)://暂停协程 x 秒后继续执行。

yield return new WaitForSeconds(2);  // 暂停 2 秒

yield return new WaitUntil(predicate)://等待直到某个条件为 true 时继续执行。

yield return new WaitUntil(() => someCondition == true);  // 等待某个条件成立

yield return StartCoroutine(otherCoroutine)://等待另一个协程执行完毕后再继续执行当前协程。

yield return StartCoroutine(OtherCoroutine());  // 等待另一个协程完成

常见用例

1. 延迟执行任务

协程经常被用来在一段时间后执行某个任务。以下是一个延迟 3 秒后执行的例子:

IEnumerator DelayedAction()
{
    yield return new WaitForSeconds(3);  // 等待 3 秒
    Debug.Log("This happens after 3 seconds");
}
2. 等待玩家输入

协程可以暂停代码执行,直到玩家做出某些输入,比如按下某个键:

IEnumerator WaitForPlayerInput()
{
    Debug.Log("Waiting for player to press Space");
    yield return new WaitUntil(() => Input.GetKeyDown(KeyCode.Space));  // 等待玩家按下空格键
    Debug.Log("Player pressed Space");
}
3. 多协程依赖执行

你可以让一个协程等待另一个协程执行完成后再继续。例如,下面的例子展示了如何等待两个协程执行完成后再执行后续逻辑:

IEnumerator Sequence()
{
    yield return StartCoroutine(Task1());
    yield return StartCoroutine(Task2());
    Debug.Log("Both tasks completed");
}

IEnumerator Task1()
{
    Debug.Log("Task 1 started");
    yield return new WaitForSeconds(2);
    Debug.Log("Task 1 completed");
}

IEnumerator Task2()
{
    Debug.Log("Task 2 started");
    yield return new WaitForSeconds(1);
    Debug.Log("Task 2 completed");
}
4. 逐帧处理任务

有时我们希望任务能在多个帧内逐渐完成,而不是一次性完成以避免卡顿。协程可以轻松实现这一点:

IEnumerator DoTaskOverTime()
{
    for (int i = 0; i < 10; i++)
    {
        Debug.Log("Processing step: " + i);
        yield return null;  // 等待到下一帧再继续执行
    }
}

协程的注意事项

        协程不是真正的多线程:

        虽然协程可以暂停执行,并在之后恢复,但它们始终在主线程中运行。也就是说,协程不会并发执行,只是通过暂停和恢复的方式避免了阻塞主线程的情况。

        Unity 的协程是始终在主线程上运行的,并不涉及多线程处理。但因为协程使用了 yield return 进行暂停,所以它可以让其他代码继续执行,而不会阻塞主线程的运行。这种非阻塞的行为有时会让人误以为协程运行在另一个线程上,但实际上它仍然依赖主线程的调度。

        假设你启动了一个协程,在协程中等待 2 秒后打印一段话,而协程下面是一个耗时的方法。

void Start()
{
    // 启动协程
    StartCoroutine(MyCoroutine());
    
    // 耗时操作
    HeavyOperation();
}

IEnumerator MyCoroutine()
{
    Debug.Log("Before waiting");
    
    // 暂停协程 2 秒
    yield return new WaitForSeconds(2);
    
    Debug.Log("After waiting 2 seconds");
}

void HeavyOperation()
{
    // 模拟一个非常耗时的操作,比如计算密集型任务
    Debug.Log("Heavy operation started");
    for (int i = 0; i < 1000000000; i++) { }  // 假设这是一个耗时的操作
    Debug.Log("Heavy operation finished");
}

这里就会出现一个问题:

        因为 HeavyOperation() 占用了大量的 CPU 时间,主线程处理其他任务(包括更新协程的进度)可能会延迟。这意味着即使 2 秒时间已经过去了,但如果 HeavyOperation() 还没执行完,协程中的 "After waiting 2 seconds" 也会推迟打印

        当协程遇到 yield return 时,执行会挂起并交出控制权,让上级调用者(通常是 Unity 的主线程)继续执行其他任务。然后,当条件满足后,协程会恢复执行,继续从挂起的地方往下执行。

        注意这个条件满足之后,说的不清晰,让我们进一步说清楚协程恢复的时机:当协程的条件满足时(例如等待时间结束),Unity 不会立即打断主线程的操作。协程的恢复是在主线程的当前帧所有任务结束之后,才会被调度执行。

        上面的例子是等待两秒,但是下面正在执行一个需要大量cpu的操作,两秒已经过去了还没有执行完,Unity并不会打断这个操作,所以这个等待秒数的行为并不是绝对靠谱的.

        这是协程与多线程的关键区别之一。协程不会在任何时间打断主线程的工作,而是等到主线程“空闲”时(例如,一帧的任务全部完成),才会恢复执行。

        在大多数情况下,协程的等待时间(比如 WaitForSeconds(2) 等待 2 秒)是靠谱的,但确实存在一些特殊情况可能导致时间的精确度不如帧数那样可靠。具体来说,帧是更为精确和可控的,而秒的等待可能会受到主线程的负载、性能瓶颈等因素的影响。

        在 Unity 中,协程的恢复是基于帧更新的。Unity 的主线程在每一帧都会处理脚本、物理、渲染等任务,如果主线程负载过重或者有一些耗时操作,帧率可能会降低(例如从 60FPS 降到 30FPS),进而影响时间的精度。

        协程的生命周期:

        如果启动协程的 MonoBehaviour 被销毁或禁用,协程会自动停止。因此,协程的生命周期依赖于启动它的对象。

如果你希望手动停止协程,可以使用 StopCoroutine() 方法。

Coroutine myCoroutine;

void Start()
{
    myCoroutine = StartCoroutine(MyCoroutine());
}

void StopMyCoroutine()
{
    if (myCoroutine != null)
    {
        StopCoroutine(myCoroutine);
    }
}

        除了 Unity 内置的 WaitForSeconds 和 WaitUntil,你还可以自定义 yield 的行为。通过继承 CustomYieldInstruction,你可以实现复杂的等待条件。

public class WaitForCondition : CustomYieldInstruction
{
    private Func<bool> condition;

    public WaitForCondition(Func<bool> condition)
    {
        this.condition = condition;
    }

    public override bool keepWaiting
    {
        get { return !condition(); }
    }
}

// 使用自定义的协程等待条件
IEnumerator CustomWaitExample()
{
    Debug.Log("Waiting for custom condition...");
    yield return new WaitForCondition(() => Time.time > 5);
    Debug.Log("Condition met, continuing...");
}

   

  • 19
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值