Unity中巧用协程和游戏对象的生命周期处理游戏重启的问题

时间:2022-09-12 14:34:44

主要用到协程(Coroutines)和游戏对象的生命周期(GameObject Lifecycle)基础知识,巧妙解决了游戏重启的问题。

关于协程,这里有篇文章我觉得写的非常好,理解起来也很容易。推荐先看这篇文章:对Unity中Coroutines的理解>>

协程简单来看分三部分:

1)启动,常用方法:StartCoroutine(IEnumerator routine) | StartCoroutine(string methodName)

2)执行,执行的函数其返回值必须为IEnumerator(迭代器)

3)停止,常用方法:StopCoroutine(string methodName) | StopCoroutine(IEnumerator routine) | StopCoroutine(Coroutine routine) | StopAllCoroutines

启动、停止基本上都是对应的方法,有Start就有Stop(StopAllCoroutines比较特殊一点,没有对应的Start)。

那么重点就在执行的函数上,只要用到协程,就不得不提关键词 yield,先看它的语法:

yield return <expression>;
yield break;

(来源:https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/keywords/yield)

怎么理解它呢?以yield return null;为例,当执行了协程函数时,遇到yield return <expression>后,本帧就暂停执行后面的语句,然后下一帧再check表达式的返回值,如果为return则继续执行,如果不为return则到下一帧继续检测 —— 这里的前提是下一帧会执行,如果对象被销毁了,下一帧就不执行了,上面说法也就不成立了。

来验证一下上面的说法:

1、如果真的是一帧执行一次,那我就设定Unity的帧率为1(即1秒只执行一次Update);

2、在Start()函数之后,启动一个协程,里面跑一个死循环 while (true),然后输出时间,从控制台上观察其输出;

设定帧率:

1、关闭系统的帧率设定,菜单 Edit –> Project Settings –> Quality

Unity中巧用协程和游戏对象的生命周期处理游戏重启的问题

在场景的主相机上挂一个脚本,在 Start()函数中修改帧率

// 修改当前的FPS
Application.targetFrameRate = ;

上面的代码也可以放在场景加载之前的函数里,比如:OnBeforeSceneLoadRuntimeMethod

[RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.BeforeSceneLoad)]
static void OnBeforeSceneLoadRuntimeMethod()
{
// todo…
    }

2、随便找一个GameObject,在其挂载脚本的Start()函数中启动协程

Unity中巧用协程和游戏对象的生命周期处理游戏重启的问题

运行之后,在控制台上可以看到如下的输出,是符合预期的。

Unity中巧用协程和游戏对象的生命周期处理游戏重启的问题

注:前三行的输出先忽略,下面会进行说明,关键看后面的输出。

Unity内置了一些yield指令(YieldInstruction):

-)WaitForSeconds

等待指定的游戏时间(游戏流逝时间可用Time.scale调整)

-)WaitForSecondsRealtime

等待指定的真实时间(现实时间不受Time.scale影响)

-)WaitForFixedUpdate

等待下一次的FixedUpdate后再执行

-)WaitForEndOfFrame

等待这一帧(Frame)绘制完但还没有显示的时间点再执行,可以用来取得绘制结果做一些事情

-)WaitUntil

待到传入的 delegate 满足条件返回 true 后再执行

-)WaitWhile

跟WaitUntil差不多,只是传入的 delegate 满足条件返回 false 后再执行,与WaitUntil返回值正好相反

开发游戏,通常我们都需要支持热更新,而腾讯开源的XLua无疑是目前最好的一个选择。既然是热更新,游戏重启我们肯定不希望像安卓那样暴力:

1、先杀掉进程;

2、再重启启动进程;

不是不可以,只是在iOS环境下这个方案就行不通了。那怎么办呢?最好当然是游戏自己实现,像启动App进程那样,先走销毁(Destroy)流程再走初始化流程(Init)。

因为C#的脚本通常都会挂载到一个场景(Scene)或者一个游戏对象(GameObject)上,而重启就势必会销毁场景,这样脚本里的函数在Destroy就不会再被执行了。这就需要一个独立的一个C#脚本,它独立于游戏的加载、销毁之外,我第一次看到这个方案时有点惊讶,绝对是奇技淫巧。

我在示例中使用的是点击按钮时,调用函数的重启方法(Restart)

Restart方法只干了一件事,new一个GameObject对象出来,当这个GameObject被实例化时,该类的Start函数会被调用,然后在Start函数中执行:销毁 –> 间隔一段时间(用上面的yield return null或者Unity提供的延时处理指令) –> 初始化 –> 销毁 new出来的GameObject对象。

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine; public class Startup : MonoBehaviour { static GameObject m_GameObject; [RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.BeforeSceneLoad)]
static void OnBeforeSceneLoadRuntimeMethod()
{
// 在场景加载之前,可以做一些与游戏无关的一些初始化工作:Lua虚拟机的启动、SDK初始化、Bugly的初始化、资源管理的初始化... Debug.Log("Startup....... Before first scene loaded");
} // Use this for initialization
private IEnumerator Start()
{
Debug.Log("Start ----------------"); // 加载一个空的场景,设置其主相机的Clear Flag为Don't Clear,然后停一帧
yield return null; // 这里可以调用销毁的相关函数,如:关闭Lua虚拟机、相关资源的释放等,然后再停一帧 yield return null; // 重新调用初始化方法
OnBeforeSceneLoadRuntimeMethod(); // 加载游戏的主场景 Destroy(m_GameObject);
m_GameObject = null;
} public static void Restart()
{
if (m_GameObject == null)
{
m_GameObject = new GameObject("Startup", typeof(Startup));
DontDestroyOnLoad(m_GameObject);
}
} }

参考资料:

[1] Unity Coroutine 使用筆記

[2] 对Unity中Coroutines的理解

[3] Unity 协程运行时的监控和优化

Unity中巧用协程和游戏对象的生命周期处理游戏重启的问题的更多相关文章

  1. Unity脚本编程之——协程(Coroutine)

    本文翻译自Unity官方文档:https://docs.unity3d.com/Manual/Coroutines.html 专有名词: Coroutine 协程 Alpha 不透明度 当你调用一个函 ...

  2. Unity 3D Framework Designing&lpar;3&rpar;——构建View和ViewModel的生命周期

    > 对于一个View而言,本质上是一个MonoBehaviour.它本身就具备生命周期这个概念,比如,Awake,Start,Update,OnDestory等.这些是非常好的方法,可以让开发者 ...

  3. Hibernate中Java对象的生命周期

    一个对象的出生源于我们的一个new操作,当我们使用new语句创建一个对象,这个对象的生命周期就开始了,当我们不在有任何引用变量引用它,这个对象就的生命就此结束,它占用的内存就可以被JVM的垃圾回收器回 ...

  4. java之hibernate之session中对象的生命周期

    1. session是用来执行对象的crud操作,并且session是对象事务工厂.session是线程级别的,所以生命周期比较短. 2.session中对象的生命周期图: 3.session中对象的 ...

  5. yield学习续:yield return迭代块在Unity3D中的应用——协程

    必读好文推荐: Unity协程(Coroutine)原理深入剖析 Unity协程(Coroutine)原理深入剖析再续 上面的文章说得太透彻,所以这里就记一下自己的学习笔记了. 首先要说明的是,协程并 ...

  6. python中线程 进程 协程

    多线程:#线程的并发是利用cpu上下文的切换(是并发,不是并行)#多线程执行的顺序是无序的#多线程共享全局变量#线程是继承在进程里的,没有进程就没有线程#GIL全局解释器锁#只要在进行耗时的IO操作的 ...

  7. 038&lowbar;go语言中的状态协程

    代码演示: package main import ( "fmt" "math/rand" "sync/atomic" "time ...

  8. 【Unity笔记】使用协程(Coroutine)异步加载场景

    using UnityEngine; using System.Collections; using UnityEngine.SceneManagement; using System; public ...

  9. Unity带参数的协程

    两种方法都可以传递参数,代码如下: using UnityEngine; using System.Collections; public class Test : MonoBehaviour { v ...

随机推荐

  1. 更改Windows控制台默认缓冲区行数和宽度

    Windows控制台窗口默认的显示行数很少, 以至于有时候要显示很多东西的时候, 总是导致前面的内容丢失. 很不爽. 于是GG了下, 在*上找到了答案~~~ 设置方式: 1. ...

  2. VS2012中启动性能分析 独占样本数的分析

    http://www.cnblogs.com/TianFang/archive/2012/11/18/2776266.html

  3. SVD学习

    前言: 上一次写了关于PCA与LDA的文章,PCA的实现一般有两种,一种是用特征值分解去实现的,一种是用奇异值分解去实现的.在上篇文章中便是基于特征值分解的一种解释.特征值和奇异值在大部分人的印象中, ...

  4. samba服务器的安装及配置

    安装前首先查看服务器是否已经安装samba服务器 [root@bogon home]# rpm -qa|grep samba system-config-samba-docs-1.0.9-1.el6. ...

  5. 玩转kindle paperwhite: 如何越狱,安装强大外挂软件koreader

    NOTICE 1: 在更新kpvbooklet和使用最新版本的koreader(v2013.03-211)时候,会出现pdf文档无法重排的错误.亲测. 如果你是使用的最新版本koreader且出现上述 ...

  6. SQL 多个表之间联合查询

    非常少用join,这次学学,并备忘两篇文章! 转自:http://hcx-2008.javaeye.com/blog/285661 连接查询 通过连接运算符能够实现多个表查询.连接是关系数据库模型的主 ...

  7. 1&period;1XAF框架开发视频教程-简单的订单管理实现过程&comma;视频&comma;提纲&comma;及教程源码

    下面是视频教程的提纲: PPT版本的提纲下载 本节源码下载 XAF框架开发教程 快速实现企业级信息系统开发的利器 XAF简介 ´  开发公司:www.devexpress.com,老牌控件公司 ´  ...

  8. adb获取Android性能数据

    环境:Android测试环境 搭建Android测试环境: 1.下载AndroidSDK: 2.配置环境变量: (1).ANDROID_HOME (2).ANDROID_HOME-TOOLS (3). ...

  9. 剑指Offer——京东校招笔试题&plus;知识点总结

    剑指Offer--京东校招笔试题+知识点总结 笔试感言 经过一系列的笔试,发觉自己的基础知识还是比较薄弱的,尤其是数据结构和网络,还有操作系统.工作量还是很大的.做到精确制导的好方法就是在网上刷题,包 ...

  10. 搭建Keil C51开发环境

    安装Keil 首先需要获取Keil的安装程序,然后双击开始安装.安装过程非常简单,基本都是一路next.具体步骤如下图: 注册 如果不注册,会有许多功能限制,例如限制生成的代码的大小.为了学习和和教学 ...