Unity之协程

协程(Coroutine)是 Unity 中一种可以暂停执行、稍后再继续 的特殊函数,它不是多线程,而是在主线程中执行的,只是通过yield关键字实现了 "分段执行" 的效果。

一、核心原理

  • 协程本质是一个返回值为IEnumerator的函数,通过yield return语句标记暂停点。
  • Unity 的主线程每帧会检查所有运行中的协程,当协程的暂停条件满足(比如等待 1 秒、等待帧结束),就会从暂停点继续执行后续代码。
  • 协程始终在主线程运行,因此不能用协程解决耗时计算(耗时操作需用多线程),但适合处理 "需要等待" 的逻辑(如等待动画播放、等待网络请求响应)。

二、协程的基础语法

1. 定义协程

协程函数必须满足:

  • 返回值为IEnumerator
  • 函数体中包含yield return语句
  • 函数名通常以StartXXX开头(约定俗成)
cs 复制代码
using UnityEngine;
using System.Collections; // 必须引入此命名空间

public class CoroutineDemo : MonoBehaviour
{
    // 定义一个简单的协程
    IEnumerator SimpleCoroutine()
    {
        Debug.Log("协程开始执行 - 第一帧");
        yield return null; // 暂停,下一帧继续执行
        
        Debug.Log("协程继续执行 - 第二帧");
        yield return new WaitForSeconds(2f); // 暂停2秒后继续
        
        Debug.Log("协程执行完毕 - 2秒后");
    }
}
2. 启动协程

通过StartCoroutine()方法启动协程,有两种方式:

cs 复制代码
void Start()
{
    // 方式1:直接传入协程函数(推荐,便于后续停止)
    StartCoroutine(SimpleCoroutine());
    
    // 方式2:传入字符串(性能稍差,只能停止该字符串对应的协程)
    StartCoroutine("SimpleCoroutine");
}
3. 停止协程

停止协程有 3 种常用方式:

cs 复制代码
// 方式1:停止指定的协程(需保存协程引用)
private Coroutine myCoroutine;
void Start()
{
    myCoroutine = StartCoroutine(SimpleCoroutine());
}
void StopMyCoroutine()
{
    StopCoroutine(myCoroutine);
}

// 方式2:停止当前脚本中所有协程
void StopAllMyCoroutines()
{
    StopAllCoroutines();
}

// 方式3:销毁挂载脚本的GameObject(协程会自动停止)
void DestroyGameObject()
{
    Destroy(gameObject);
}

三、常用的 yield 返回值(暂停条件)

协程的核心是yield return后的暂停条件,以下是最常用的类型:

|---------------------------------|------------------------------|--------------------------------------------------|
| 返回值 | 作用 | 示例 |
| null | 暂停 1 帧,下一帧继续 | yield return null; |
| WaitForSeconds(float) | 等待指定秒数(受 Time.timeScale 影响) | yield return new WaitForSeconds(1.5f); |
| WaitForSecondsRealtime(float) | 等待指定秒数(不受 Time.timeScale 影响) | yield return new WaitForSecondsRealtime(1.5f); |
| WaitForEndOfFrame() | 等待当前帧所有渲染完成后继续 | yield return new WaitForEndOfFrame(); |
| WaitForFixedUpdate() | 等待下一次 FixedUpdate 执行后继续 | yield return new WaitForFixedUpdate(); |
| WWW /UnityWebRequest | 等待网络请求完成(Unity 旧 / 新接口) | yield return webRequest.SendWebRequest(); |
| AnotherCoroutine() | 等待另一个协程执行完毕 | yield return AnotherCoroutine(); |

案例 :分步执行动画 / 逻辑

cs 复制代码
using UnityEngine;
using System.Collections;

public class StepByStepLogic : MonoBehaviour
{
    void Start()
    {
        StartCoroutine(ExecuteStepByStep());
    }

    IEnumerator ExecuteStepByStep()
    {
        // 第一步:播放开门动画
        Debug.Log("播放开门动画");
        yield return new WaitForSeconds(2f); // 等待动画播放完毕
        
        // 第二步:移动角色到指定位置
        Debug.Log("移动角色");
        yield return new WaitForSeconds(1.5f);
        
        // 第三步:播放拾取物品动画
        Debug.Log("拾取物品");
        yield return new WaitForSeconds(1f);
        
        // 第四步:完成所有操作
        Debug.Log("任务完成");
    }
}
相关推荐
叶帆14 天前
【YFIOs】用C#开发硬件之设备上云
开发语言·unity·c#
久数君14 天前
AI三维建模工具“造形家”:地理场景三维化的高效解决方案
unity·glb·ai算法·ai三维建模工具·地图框选·造形家·城市建筑模型
会思考的猴子15 天前
Unity VFX 属性 Postion 和 TargetPostion
unity
hai31524754315 天前
九章编程法 · 猜数字游戏 (GW-BASIC 重构版) *
人工智能·microsoft·游戏引擎·游戏程序
心前阳光15 天前
Unity资源导入之自动化资源导入
unity·自动化·游戏引擎
心前阳光15 天前
Unity之2021.3.45f2c1发布安卓程序遇到的问题
android·unity·游戏引擎
纪纯15 天前
PicoVR Unity Integration SDK 3.4 常用交互API
unity·游戏引擎·vr·pico
龙智DevSecOps解决方案15 天前
3A 游戏优化技术栈:如何打通引擎级分析工具与 DevOps 持续集成管线?
unity·性能优化·游戏开发·技术美术·perforce·unrealengine
葛兰岱尔15 天前
从 SolidWorks 到 Three.js,从 Inventor 到 Unity——制造业CAD模型“几何-语义一体化“转换,不再是天方夜谭!
开发语言·javascript·unity
鼎艺创新科技15 天前
三维电子沙盘中OSGB倾斜摄影数据的加载与渲染
游戏引擎·cocos2d