Unity协程机制详解
Unity的协程(Coroutine)是一种异步编程的机制,允许在多个帧之间分割代码的执行,而不阻塞主线程。与传统的多线程不同,Unity的协程在主线程中运行,并不会开启新的线程。
什么是协程?
协程是一种能在一定条件下暂停执行,并在稍后恢复执行的函数。它允许将一个任务拆分为多个小任务,每一小段任务可以在多个帧内完成,而不是在一帧内阻塞主线程。
在Unity中,协程通过StartCoroutine方法启动,并可以通过yield关键字来暂停执行,直到满足特定条件(例如等待时间、等待帧、等待另一个协程完成等)后继续运行。
协程的语法
在Unity中,协程的基本语法如下:
using UnityEngine;
using System.Collections;public class CoroutineExample : MonoBehaviour
{void Start(){// 启动一个协程StartCoroutine(MyCoroutine());}IEnumerator MyCoroutine(){Debug.Log("第1步: 开始协程");// 等待3秒yield return new WaitForSeconds(3);Debug.Log("第2步: 等待3秒完成");// 等待一帧yield return null;Debug.Log("第3步: 等待一帧完成");// 等待直到某个条件为真yield return new WaitUntil(() => Time.time > 5);Debug.Log("第4步: 等待直到Time.time > 5");}
}
- StartCoroutine(MyCoroutine()):启动一个协程,执行MyCoroutine()函数。
- yield return new WaitForSeconds(3):暂停执行,等待3秒后恢复执行。
- yield return null:暂停当前协程,等到下一帧再恢复执行。
- yield return new WaitUntil(() => Time.time > 5):暂停,直到Time.time > 5为true时恢复执行。
启动和停止协程
启动协程
StartCoroutine(“方法名”):使用字符串指定协程名称。
StartCoroutine(方法()):传递协程的IEnumerator方法。
示例:
StartCoroutine("MyCoroutine");
StartCoroutine(MyCoroutine());
停止协程
StopCoroutine(方法名或引用):停止指定的协程。
StopAllCoroutines():停止脚本中所有正在运行的协程。
Coroutine myCoroutine;void Start()
{myCoroutine = StartCoroutine(MyCoroutine());
}void StopIt()
{StopCoroutine(myCoroutine); // 停止指定的协程StopAllCoroutines(); // 停止当前脚本中的所有协程
}
注意:使用StopCoroutine时,不能使用StartCoroutine(“MyCoroutine”)的字符串方式,因为无法引用该协程。
协程的yield语法详解
yield的作用是暂停当前协程的执行,直到指定的条件满足。Unity中常见的yield操作有:
- yield return null; 等待一帧
- yield return new WaitForSeconds(t); 等待t秒
- yield return new WaitUntil(predicate); 等待直到条件为真
- yield return new WaitWhile(predicate); 等待直到条件为假
- yield return StartCoroutine(协程); 等待另一个协程完成
- yield break; 终止协程,立即退出
常见的协程应用场景
场景切换中的加载动画
在场景加载的过程中显示加载动画,而不是卡顿的黑屏。
IEnumerator LoadSceneAsync()
{AsyncOperation operation = SceneManager.LoadSceneAsync("SceneName");while (!operation.isDone){Debug.Log($"加载进度:{operation.progress * 100}%");yield return null; // 等待一帧}Debug.Log("场景加载完成");
}
动画和特效的控制
在播放一段动画或特效时,等待动画播放完成再继续后续操作。
IEnumerator PlayAnimation()
{Animator animator = GetComponent<Animator>();animator.Play("Attack");yield return new WaitForSeconds(2); // 等待动画播放2秒Debug.Log("动画播放完成,执行其他操作");
}
定时触发的任务
IEnumerator SpawnEnemies()
{while (true){Instantiate(enemyPrefab, transform.position, Quaternion.identity);yield return new WaitForSeconds(5); // 每隔5秒刷怪}
}
协程和多线程的区别
协程 | 多线程 |
---|---|
运行在主线程中 | 每个线程有独立的执行流 |
使用yield暂停 | 使用Thread.Sleep等 |
不需要上下文切换,效率高 | 需要操作系统的线程调度,性能开销大 |
主要用于异步逻辑操作 | 主要用于计算密集型操作 |
注意事项和最佳实践
- 不要阻塞主线程:协程在主线程中运行,while(true)的无限循环会冻结游戏。
- 避免频繁调用StartCoroutine:如果在Update中频繁调用StartCoroutine,会导致性能下降。
- 避免内存泄漏:未手动停止的协程会一直运行,导致内存泄漏。可用StopCoroutine来手动控制协程的停止。
- 不要用字符串调用协程:StartCoroutine(“MyCoroutine”)的方式不易调试和管理,建议使用StartCoroutine(MyCoroutine())。
- 不要滥用协程:协程适用于需要在多个帧内分段执行的任务。对于简单的帧内任务,尽量不要使用协程。
Unity的协程机制提供了一种轻量的异步任务调度,适合在游戏中实现等待、计时、加载动画、场景切换等操作。与多线程相比,协程的运行成本更低,控制也更简单。通过yield来暂停和恢复代码执行,可以显著提高游戏性能和玩家体验。合理使用协程有助于优化游戏中的异步任务。
Unity是如何实现协程的
Unity的协程原理基于迭代器(Iterator)和C#的IEnumerator接口,而不是多线程。Unity的协程并不会新开一个线程去执行逻辑,而是在主线程中调度,通过“分段执行”的方式控制代码的暂停和恢复。
协程的原理概述
- C#中的迭代器和yield return
- 协程的本质是C#的迭代器(Iterator),而yield return就是一种控制流的中断标志。
- 当执行到yield return时,Unity会将当前协程的“状态”保存起来,并在下一个“可用的时机”恢复执行。
- Unity的调度器(Coroutine Scheduler)
- Unity每一帧都会检查哪些协程需要恢复执行。
- 如果一个协程调用了yield return WaitForSeconds(2),Unity会在2秒后恢复该协程,并从上次中断的位置继续执行。
- 这种调度机制与多线程无关,一切都在主线程中运行。
- 执行流程
- Unity的MonoBehaviour管理协程,通过内部的队列/列表维护所有活跃的协程。
- 每一帧,Unity会遍历这些协程,检查是否满足“恢复执行”的条件(如时间已到、等待的条件达成等)。
- 如果条件满足,Unity会调用该协程的MoveNext()方法,让协程继续运行,直到遇到下一个yield语句。
协程的实现细节
要理解Unity如何实现协程,必须先掌握C#的IEnumerator和yield return的机制。
- C# 的 IEnumerator 和 yield return
当在C#中使用yield return时,编译器会自动将方法“拆分”成状态机(State Machine),而这个状态机可以跟踪函数的当前执行位置。
示例如下:
IEnumerator MyCoroutine()
{Debug.Log("第1步");yield return new WaitForSeconds(1); // 暂停1秒Debug.Log("第2步");yield return new WaitForSeconds(2); // 暂停2秒Debug.Log("第3步");
}
编译器生成的等价状态机代码如下(简化版,省略了复杂的状态机细节):
public class MyCoroutineStateMachine : IEnumerator
{private int state = 0; // 用于跟踪当前执行到的位置public object Current { get; private set; } // yield返回的内容public bool MoveNext(){switch (state){case 0:Debug.Log("第1步");Current = new WaitForSeconds(1);state = 1;return true; // 表示协程还没有结束case 1:Debug.Log("第2步");Current = new WaitForSeconds(2);state = 2;return true;case 2:Debug.Log("第3步");state = -1; // 表示协程结束return false;}return false;}public void Reset() { }
}
解释:
- state:控制当前协程的“步骤状态”,state=0表示正在执行第1步,state=1表示执行第2步。
- Current:表示yield return的结果(如WaitForSeconds(1))。
- MoveNext():每当Unity的调度器在新一帧时调用MoveNext(),协程的控制流就会跳转到当前的state位置。
- 当state为-1时,协程结束,MoveNext()返回false。
这段代码表明,Unity通过“状态机 + 迭代器”的机制,将协程拆分成多个小的执行步骤,这些步骤会在每一帧内被调度器分阶段运行。
- Unity的调度器 (Scheduler)
- Unity在每一帧都会遍历所有的协程,并调用每个协程的MoveNext()方法。
- 如果MoveNext()返回true,Unity会继续等待,否则Unity将把这个协程从队列中删除。
- Unity在执行MoveNext()时会检查Current的返回值:
- 如果是null:表示立即恢复(继续执行)
- 如果是WaitForSeconds:Unity会记录当前的“等待时间”,然后延迟恢复协程。
- 如果是CustomYieldInstruction:例如yield return new WaitUntil(),Unity会检测其条件是否满足。
Unity的协程工作流程图
[Unity Start] ↓
[协程被添加到队列] ↓
[每一帧调用协程的MoveNext()] ↓
[检测Current] └─> 如果是 WaitForSeconds,等待指定时间└─> 如果是 WaitUntil/WaitWhile,检查条件└─> 如果是 null,直接继续执行↓
[如果协程结束,则从队列中移除]
小结
Unity的协程依赖C#的迭代器,通过IEnumerator生成的状态机来分段执行代码。
Unity的调度器每一帧都会检查和恢复协程,基于Current的结果来决定是否继续执行。
协程不会并行执行,只是在主线程上“暂停-恢复”代码。
相关文章:
Unity协程机制详解
Unity的协程(Coroutine)是一种异步编程的机制,允许在多个帧之间分割代码的执行,而不阻塞主线程。与传统的多线程不同,Unity的协程在主线程中运行,并不会开启新的线程。 什么是协程? 协程是一种…...
2024年【高压电工】最新解析及高压电工考试总结
高压电工考试是电力行业从业人员必须通过的资格考试之一,它不仅检验了考生对高压电技术的掌握程度,还考验了考生在实际操作中的安全意识和应急处理能力。为了帮助广大考生更好地备考,本文整理了10道2024年高压电工考试的最新解析及总结试题&a…...
OELOVE 6.0城市列表模板
研究了好久OELOVE6.0源码,一直想将城市列表给单独整出来,做地区排名,但是PHP程序都是加密的,非常难搞,做二开都是要命的处理不了,在这里有一个简单方法可以处理城市列表,并且可以自定义TDK&…...

如何将你的 Ruby 应用程序从 OpenSearch 迁移到 Elasticsearch
作者:来自 Elastic Fernando Briano 将 Ruby 代码库从 OpenSearch 客户端迁移到 Elasticsearch 客户端的指南。 OpenSearch Ruby 客户端是从 7.x 版 Elasticsearch Ruby 客户端分叉而来的,因此代码库相对相似。这意味着当将 Ruby 代码库从 OpenSearch 迁…...

day1数据结构,关键字,内存空间存储与动态分区,释放
小练习 在堆区空间连续申请5个int类型大小空间,用来存放从终端输入的5个学生成绩,然后显示5个学生成绩,再将学生成绩升序排序,排序后,再次显示学生成绩。显示和排序分别用函数完成(两种排序方法࿰…...

1_linux系统网络性能如何优化——几种开源网络协议栈比较
之前合集《计算机网络从入门到放弃》第一阶段算是已经完成了。都是理论,没有实操,让“程序猿”很难受,操作性不如 Modbus发送的报文何时等到应答和 tcp通信测试报告单1——connect和send。开始是想看linux内核网络协议栈的源码,然…...

【问题记录】07 MAC电脑,使用FileZilla(SFTP)连接堡垒机不成功
项目场景: 使用MAC电脑,以子账号(非root)的形式登录,连接堡垒机CLB(传统型负载均衡),使用FileZilla(SFTP)进行FTP文件传输。 问题描述: MAC电脑…...

前端报错npm ERR cb() never called问题
环境使用node版本v14.21.3,npm版本6.14.18 1.问题描述 1.1使用npm install后报错 npm ERR! cb() never called!npm ERR! This is an error with npm itself. Please report this error at: npm ERR! ? ? <https://npm.community>npm ERR! A complete log…...

康谋方案 | 多源相机数据采集与算法集成测试方案
目录 一、相机组成 二、多源相机采集与测试方案 三、应用案例分享 四、结语 在智能化技术快速发展当下,图像数据的采集与处理逐渐成为自动驾驶、工业等领域的一项关键技术。高质量的图像数据采集与算法集成测试都是确保系统性能和可靠性的关键。随着技术的不断进…...

Graspness 端到端抓取点估计 | 环境搭建 | 模型推理测试
在复杂场景中实现抓取检测,Graspness是一种端到端的方法; 输入点云数据,输出抓取角度、抓取深度、夹具宽度等信息。 开源地址:https://github.com/rhett-chen/graspness_implementation?tabreadme-ov-file 论文地址࿱…...
交换机是如何避免数据碰撞的(详细解释 + 示例)
交换机是如何避免数据碰撞的(详细解释 示例) 1. 独立冲突域 交换机的每个端口都形成一个独立的冲突域。这意味着通过交换机连接的每个设备都拥有自己的通信通道,互不干扰。 示例: 假设一个交换机有4个端口,分别连接…...
魅族手机刷官方系统
从魅族官网下载固件 https://flyme.cn/firmware.html 找到自己的型号,里面有历史版本、最新版,按照需求下载。 下载的是update.zip,改名就不能升级了 方法1 直接点击下载的update.zip包就可以升级。 方法2 将文件移动到文件管理的根目录&a…...
女人想要的,是那份懂她的情绪价值
女人想要的,是那份懂她的情绪价值 在情感的世界里,我们常常听到这样的声音:“我不需要你帮我解决问题,我只希望你能懂我。”这句话,简单却深刻,它揭示了女性在情感需求上的一个独特面向——她们渴望的&…...
[python SQLAlchemy数据库操作入门]-10.性能优化:提升 SQLAlchemy 在股票数据处理中的速度
哈喽,大家好,我是木头左! 当处理大量数据时,如股票数据,默认的ORM操作可能会显得效率低下。本文将探讨如何通过一些技巧和策略来优化SQLAlchemy ORM的性能,从而提升其在股票数据处理中的速度。 1. 选择合适的数据类型 在定义模型时,选择合适的数据类型对于性能至关重要…...

【网络取证篇】取证实战之PHP服务器镜像网站重构及绕密分析
【网络取证篇】取证实战之PHP服务器镜像网站重构及绕密分析 在裸聊敲诈、虚假理财诈骗案件类型中,犯罪分子为了能实现更低成本、更快部署应用的目的,其服务器架构多为常见的初始化网站架构,也称为站库同体服务器!也就是说网站应用…...
[python]使用 Pandas 处理 Excel 数据:分割与展开列操作
在数据处理的过程中,时常需要对 Excel 表格中的数据进行清洗与转换,下面介绍使用 Python 中的 Pandas 库对 Excel 文件中的数据进行操作,具体包括分割列、展开数据、清除空格以及格式转换等操作。 目标: 读取一个没有表头的 Exc…...
单片机的选择因素
在选择单片机型号时,需要根据具体的应用需求来选择合适的单片机。单片机(Microcontroller Unit, MCU)是一种将计算机的主要部分集成在一个芯片上的微型计算机,它通常包括处理器、存储器、输入/输出接口等。随着技术的发展…...
软件测试兼容性测试丨分布式测试与多设备管理
本文将从分布式测试的概念、重要性以及实施方法入手,紧接着探讨多设备管理的必要性和管理策略,最后分析其对软件测试行业的前景与影响。 分布式测试简介 什么是分布式测试? 分布式测试是指将测试任务分散到不同的计算机或者设备上进行&…...

Linux驱动开发(13):输入子系统–按键输入实验
计算机的输入设备繁多,有按键、鼠标、键盘、触摸屏、游戏手柄等等,Linux内核为了能够将所有的输入设备进行统一的管理, 设计了输入子系统。为上层应用提供了统一的抽象层,各个输入设备的驱动程序只需上报产生的输入事件即可。 下…...

微服务篇-微服务保护:使用 Sentinel 来实现请求限流、线程隔离、服务熔断和 Fallback 备用方案的使用
🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 微服务保护 1.1 请求限流方案 1.2 线程隔离方案 1.3 服务熔断方案 2.0 Sentinel 2.1 Sentinel 安装 2.2 微服务整合 3.0 Sentinel-请求限流 4.0 Sentinel-线程隔离…...

解决pycharm同一个文件夹下from *** import***仍显示No module named
1、,from ***import *,同文件夹中已有.py文件但是仍然报错No module named 原因是因为pycharm没有把文件夹设置为根目录,只需要在文件夹的上一级设置为根目录即可,测试过如果仅仅将当前的文件夹设置仍然报错,如果把最上…...

从“人找政策”到“政策找人”:智能退税ERP数字化重构外贸生态
离境退税新政核心内容与外贸企业影响 (一)政策核心变化解析 退税商店网络扩容 新政明确鼓励在大型商圈、旅游景区、交通枢纽等境外旅客聚集地增设退税商店,并放宽备案条件至纳税信用M级企业。以上海为例,静安区计划新增1000家退…...
【C/C++】std::vector成员函数清单
文章目录 std::vector使用指南1 不同版本提供的能力基础:C98 / C03 提供的成员函数C11 新增的成员函数C14:基本无变化(主要是标准库泛化,非 vector 成员变化)C17 引入的新特性(间接影响)C20 新增…...
Java数据校验:确保数据完整性和正确性
在软件开发中,数据校验是确保应用程序数据完整性和正确性的关键步骤。Java 提供了多种方式来实现数据校验,从简单的条件检查到复杂的框架支持。在这篇博客中,我们将探讨 Java 中数据校验的重要性、常用的校验注解以及如何整合校验框架来提高代…...
如何判断指针是否需要释放?
在 C 中判断一个指针是否需要释放可以考虑以下几个方面: 一、确定指针的来源 1. 动态分配的内存: 如果指针是通过new、new[]、malloc、calloc等动态内存分配函数获取的,那么在不再需要该内存时,必须手动释放。 例如:…...

前端vue3 上传/导入文件 调用接口
点击按钮导入: <el-uploadaction"https://run.mocky.io/v3/9d059bf9-4660-45f2-925d-ce80ad6c4d15":auto-upload"false":on-change"handleFileChange":show-file-list"false"><el-button type"warning"…...

第六十二节:深度学习-加载 TensorFlow/PyTorch/Caffe 模型
在计算机视觉领域,OpenCV的DNN(深度神经网络)模块正逐渐成为轻量级模型部署的利器。本文将深入探讨如何利用OpenCV加载和运行三大主流框架(TensorFlow、PyTorch、Caffe)训练的模型,并提供完整的代码实现和优化技巧。 一、OpenCV DNN模块的核心优势 OpenCV的DNN模块自3.3…...

固定ip和非固定ip的区别是什么?如何固定ip地址
在互联网中,我们常会接触到固定IP和非固定IP的概念。它们究竟有何不同?如何固定IP地址?让我们一起来探究这个问题。 一、固定IP和非固定IP的区别是什么 固定IP(静态IP)和非固定IP(动态IP)是两种…...

全流程开源!高德3D贴图生成系统,白模一键生成真实感纹理贴图
导读 MVPainter 随着3D生成从几何建模迈向真实感还原,贴图质量正逐渐成为决定3D资产视觉表现的核心因素。我们团队自研的MVPainter系统,作为业内首个全流程开源的3D贴图生成方案,仅需一张参考图与任意白模,即可自动生成对齐精确…...

【计算机网络】Linux下简单的TCP服务器(超详细)
服务端 创建套接字 💻我们将TCP服务器封装成一个类,当我们定义出一个服务器对象后需要马上对服务器进行初始化,而初始化TCP服务器要做的第一件事就是创建套接字。 TCP服务器在调用socket函数创建套接字时,参数设置如下࿱…...