1.17 从0开始学习Unity游戏开发--场景切换
前面的所有文章我们都在一个固定的游戏场景内进行开发,在最开始介绍场景这个概念的时候就已经提及,这个场景可以是一张地图,或者是一个对战房间等等,所以显然这个场景可以有多个,并且可以从一个场景切换到另外一个场景,那么在Unity中如何进行场景切换,以及如何处理好场景切换时的各个逻辑呢,本章就会详细讲解。
新建第二个场景
还记得最早讲的如何创建场景资源吗?
在Project窗口里面随便哪个你喜欢的位置右键Create->Scene就可以创建一个新的场景资源,我们已经有了一个Demo场景,那么我们创建一个新的场景叫AnotherDemo
OK,接下来我们需要编辑这个场景的内容,那就是双击这个场景资源文件,注意如果当前打开的场景比如你现在打开的是Demo,然后你有一些修改没有保存(比如Demo的Hierarchy窗口里面根节点是带星号的),那么Unity会提示你要不要保存,你自己决定要不要保存,然后才会打开刚刚双击的场景文件。
双击过后,我们就进入到了这个场景里面,一切都回到了原点:
唯独不同的是,Hierarchy窗口的根节点换成了我们新创建的名字AnotherDemo。
ok,这样就理解了如何创建新的场景,并进去编辑内容,我们先换回到Demo场景中,因为我们接下来需要从Demo场景通过代码逻辑切换到AnotherDemo。
场景切换
既然我们要切换场景,肯定是要有什么逻辑触发,我们可以单独写个组件加到空的GameObject上,组件就在Start触发后3秒进行场景切换,而场景切换的API则是:
https://docs.unity3d.com/ScriptReference/SceneManagement.SceneManager.LoadScene.html
我们让gpt帮我们写个:
using UnityEngine;
using UnityEngine.SceneManagement;public class SceneSwitcher : MonoBehaviour
{public string sceneName;public float delay = 3f;private float startTime = 0f;private bool isSwitching = false;private void Start(){startTime = Time.time;}private void Update(){if (!isSwitching){float elapsedTime = Time.time - startTime;if (elapsedTime >= delay){SceneManager.LoadScene(sceneName);isSwitching = true;}}}
}
写的还不错,delay表示物体开始跑起来后等待多少秒执行场景切换,sceneName则是我们的场景资源文件的名字,Unity相当于收集了我们所有场景文件资源,所以可以通过直接传入名字来切换,聪明的你肯定会想到,如果重名了怎么办,毕竟文件在不同的文件夹下可以拥有一样的名字,Unity的API也给出了详细的说明:
The givensceneName
can either be the Scene name only, without the.unity
extension, or the path as shown in the BuildSettings window still without the.unity
extension. If only the Scene name is given this will load the first Scene in the list that matches. If you have multiple Scenes with the same name but different paths, you should use the full path.
Note that sceneName is case insensitive, except when you load the Scene from an AssetBundle.
说的就比较清楚,不要带文件名后缀,我们知道场景资源文件实际在操作系统下的文件名后缀是.unity,我们不需要传入这个后缀,例如就是AnotherDemo,不需要AnotherDemo.unity,如果只传了个名字,会使用第一个找到的,如果传的是路径,则按照路径去找,而路径则是Assets文件夹下的路径,例如Scenes/SampleScene(当然我们创建的并没有在这个文件夹下面)
最后注意这里的路径或者名字是大小写不敏感的,除非是从资源包里面加载,至于资源包是啥,我们后面资源加载的篇章再来聊。
那么我们这里sceneName可以有两种填法:
- 直接填AnotherDemo
- 填路径AnotherDemo
这两个都没问题,但是我们先创建一个物体把我们的组件挂上去:
Ok,我们跑起来看看,哎,为什么没切换呢?不要着急,有没有注意到编辑器界面底部有一条红色的错误信息?打开Console窗口查看全部的输出:
其实可以看到,说的就是我们需要切换的场景AnotherDemo并没有加入到build settings中,我们在游戏打包的那篇中,有提到过如何将场景加入到我们的打包内容中,其实就是这个操作。
我们从Unity编辑器窗口顶部的菜单File->Build Settings打开:
我们确实没有把AnotherDemo加入进去,那我们还是按照老办法,打开AnotherDemo这个场景文件,然后点击上面这个窗口的右边按钮Add Open Scenes,把场景加进去:
Ok,我们重新打开Demo场景,然后再跑一下,等待3秒后,确实切换到AnotherDemo中一篇虚无的世界了。
场景切换的物体保持
我们成功的切换了之后,可以很显然的理解,旧场景里面的所有GameObject都被自动销毁了,这很好,但是有一些GameObject我们是希望跨场景的,最常见的就是UI,很显然我们不能因为切换了一个场景就导致我们的UI全部消失。
那么我们需要将需要在场景切换后仍然保持存在的GameObject使用特殊API进行标记:
Object.DontDestroyOnLoaddocs.unity3d.com/ScriptReference/Object.DontDestroyOnLoad.html
没错,就是这么简单,我们只需要用这个DontDestoryOnLoad来传入我们需要保持的GameObject即可。我们修改一下场景切换的脚本,允许我们设置哪些GameObject可以被保持:
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.SceneManagement;public class SceneSwitcher : MonoBehaviour
{public string sceneName;public float delay = 3f;public List<GameObject> gameObjectToStayAlive;private float startTime = 0f;private bool isSwitching = false;private void Start(){startTime = Time.time;}private void Update(){if (!isSwitching){float elapsedTime = Time.time - startTime;if (elapsedTime >= delay){if (gameObjectToStayAlive != null){foreach (GameObject go in gameObjectToStayAlive){DontDestroyOnLoad(go);}}SceneManager.LoadScene(sceneName);isSwitching = true;}}}
}
可以看到,我们新增了一个GameObject数组gameObjectToStayAlive来存放我们希望切换场景时会继续带到下个场景的GameObject,在切换场景之前我们通过DontDestroyOnLoad来设置这些GameObject不要因为场景切换而销毁。
然后我们可以看到编辑器面板上就能通过点击数组的加减符号来新增一个数组元素:
这其实是Unity编辑器默认给数组成员画的一个更方便一点的编辑方法,如果用过更老一些版本的Unity应该能知道以前的编辑方法很落后,不好用。
这里我们将UI显示所需的元素加进去,注意父元素设置DontDestroyOnLoad对子元素也是生效的,所以我们这里没必要将Canvas物体下的所有子元素都加进去。
Ok,那我们跑起来看看?
切换场景后,我们看到UI还能生效,并且可以观察到,我们设置过的GameObject都跑到了一个奇怪的地方:
他们都跑到了一个DontDestroyOnLoad下面,这其实就是Unity实现DontDestroyOnLoad方案,也就是我们实际打开了两个场景,一个是AnotherDemo场景,一个是DontDestroyOnLoad场景,这两个场景同时生效,而我们设置了DontDestroyOnLoad的GameObject会待在DontDestroyOnLoad场景内,所以即使我们切换其他场景也不会影响在DontDestroyOnLoad场景内的物体的销毁逻辑。
不过现在我们发现了另外一个问题,Console窗口里面疯狂报错:
看起来都是同一个错误,仔细看一下,其实就能理解,这个是我们的UI准心上,挂了一个AimController组件,这个组件有引用场景中的相机来确定我们开火的方向,而我们切换场景了,之前引用的Demo场景里面的相机已经不复存在,那么我们继续引用的话,肯定就报错了。同理的还有FireController组件的引用也是一样,我们切换了场景后,这个被引用的组件所在的GameObject也已经销毁了。
所以我们要解决这个问题的话,就需要在切换场景的时候清理掉旧的引用,并且做好判空的逻辑:
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.SceneManagement;public class SceneSwitcher : MonoBehaviour
{public string sceneName;public float delay = 3f;public List<GameObject> gameObjectToStayAlive;public AimController aimController;private float startTime = 0f;private bool isSwitching = false;private void Start(){startTime = Time.time;}private void Update(){if (!isSwitching){float elapsedTime = Time.time - startTime;if (elapsedTime >= delay){if (gameObjectToStayAlive != null){foreach (GameObject go in gameObjectToStayAlive){DontDestroyOnLoad(go);}}if (aimController != null){aimController.mainCamera = null;aimController.fireController = null;}SceneManager.LoadScene(sceneName);isSwitching = true;}}}
}
我们朴素的加了一个aimController成员,然后期望有人能在编辑器里面赋值好这个组件,这样我们在场景切换的时候就能够顺利的清理掉带不走的引用。
同理,我们在AimController脚本里面也需要对两个成员进行判空处理:
using UnityEngine;public class AimController : MonoBehaviour
{public Camera mainCamera;public FireController fireController;private RectTransform rectTransform;private void Start(){rectTransform = GetComponent<RectTransform>();}private void Update(){rectTransform.anchoredPosition = Input.mousePosition;if (mainCamera == null || fireController == null){return;}// 获取屏幕上当前鼠标位置(也就是准心位置)所在的3D空间位置Vector3 aimWorldPosition = mainCamera.ScreenToWorldPoint(new Vector3(Input.mousePosition.x,Input.mousePosition.y, mainCamera.nearClipPlane));// 通过坐标相减可以得到方向向量Vector3 fireDirection = aimWorldPosition - mainCamera.transform.position;// 归一化后传递给开火控制脚本fireController.SetDirection(fireDirection.normalized);}
}
Ok,那我们在场景里面赋值一下aimController:
再跑一下,错误消失。
从这里我们就可以看到,场景切换保持物体的存在虽然用起来很简单,但是实际上我们仍然需要小心的管理所有对于旧场景的引用,并且所有设置了DontDestroyOnLoad的物体,已经不会因为切换场景而销毁,所以理所当然的需要我们自己管理什么时候去Destroy它。
思考题
- 我们切换场景后,抛弃了旧场景里面的相机和FireController,但是我们在新场景里面肯定也是希望可以瞄准+射击的,新场景里面已经有了一个自带的Main Camera,所以现在我们该怎么做来让切换场景后也能正常射击?
- 在上面代码里面,我们简单的加入了一个AimController来实现清理旧引用的逻辑,但是作为一个合格的程序员肯定第一时间想到的是职责单一的设计原则,所以想想看,一个合理的解耦写法应该是如何?
- 在切换场景的API的官方说明里面,其实已经有明确的提示,这个切换API是阻塞同步操作,应该使用异步API,那么如果使用异步API,我们又应该如何改写代码呢?
下一章
本章我们详细的了解了一下如何切换场景以及切换场景如果要保持GameObject存活要怎么做,以及这么做了之后遇到的一些现实问题。
切换场景的时候其实已经从官方文档中开始了解到有AssetBundle这样的字眼,其实这就是Unity在正式游戏包中的资源管理办法,当我们需要动态加载资源而不是像现在都是引用来搞定的时候,就需要动态加载资源了,所以下一章我们将会讲解Unity内部动态加载资源的几种办法和适用场景。
相关文章:

1.17 从0开始学习Unity游戏开发--场景切换
前面的所有文章我们都在一个固定的游戏场景内进行开发,在最开始介绍场景这个概念的时候就已经提及,这个场景可以是一张地图,或者是一个对战房间等等,所以显然这个场景可以有多个,并且可以从一个场景切换到另外一个场景…...

【golang学习笔记】——(五)Go格式化统一代码风格
我们在入职一家新公司的时候,除了要学习公司的流程规范和规章制度,还会做的一件事情就是进行公司编码规范的学习,基于google的C规范下,做了各自的发散和规范,久而久之就是包罗万象的样子,疲于应付各种规范约…...

CAD转SHP最好的方法 赶快收藏起来吧
1、利用 ArcToolsbox 工具先将 DWG 文件转为 MDB 通过 CASS 软件生成的 DWG 文件,字段中包含有很多属性内容,所以我们先将 DWG 格式 的文件转换为 MDB 格式,再通过 MDB 转换为 SHP 格式数据进行整理。具体步骤如下: 通过 ArcTool…...

PyQt PyQt5 Python VTK Gui Actor 选中 高亮显示 actor
前言: 本文主要介绍了如何使用Python VTK高亮显示actor,使用Python语言,高亮显示选中的actor。当窗口中的圆球actor被选中时,会变成红色,并且会显示actor三遍面片边缘信息。 效果: VTK VTK,&…...
TCP和UDP通信对比
tcp通信流程 服务器: 创建流式套接字 绑定 监听 提取 读写 关闭 客户端: 创建流式套接字 连接 读写 关闭 收发数据: read recv ssize_t recv(int sockfd, void *buf, size_t len, int flags); //flagsMSG_PEEK 读数据不会删除缓冲区的数据 write send ssize_t send(int…...

SpringCloud:ElasticSearch之自动补全
当用户在搜索框输入字符时,我们应该提示出与该字符有关的搜索项,如图: 这种根据用户输入的字母,提示完整词条的功能,就是自动补全了。 因为需要根据拼音字母来推断,因此要用到拼音分词功能。 1.拼音分词器…...

TOOM解析如何搭建一套适合自己的舆情监测系统?完整的实战指南
随着互联网的普及和社交媒体的盛行,人们在网络上的活动越来越多,同时也涌现出大量的信息和舆情。这些信息和舆情在一定程度上会影响社会和个人的发展和进步。因此,舆情监测逐渐成为一项重要的任务。在本篇文章中,我们将为大家介绍…...

技术分享 | OceanBase 手滑误删了数据文件怎么办
作者:张乾 外星人2号,现兼任六位喵星人的资深铲屎官。 本文来源:原创投稿 *爱可生开源社区出品,原创内容未经授权不得随意使用,转载请联系小编并注明来源。 手滑误删了数据文件,并且没有可替换的节点时&…...

windows上Git Bash支持常用命令gcc tree zip wget cmake ninja
windows上Git Bash支持常用命令gcc tree zip wget cmake ninja 前言 Git Bash基于MinGW64, 提供了win32下的linux命令环境,如ls、cat、tar等。 但是Git Bash还是缺少一些命令,如gcc、make、tree、zip、wget、cmake、ninja等 1. Git Bash支持其他命令…...

面试题30天打卡-day10
1、String 和 StringBuffer、StringBuilder 的区别是什么? String、StringBuffer、StringBuilder主要的区别在于执行效率和线程安全上。 String:String字符串常量,意味着它是不可变的,导致每次对String都会生成新的String对象&a…...

【python】制作一个简单的界面,有手就行的界面~
目录 前言准备工作试手小案例开始我们今天的案例教学尾语 💝 前言 嗨喽~大家好呀,这里是魔王呐 ❤ ~! ttkbootstrap 是一个基于 tkinter 的界面美化库, 使用这个工具可以开发出类似前端 bootstrap 风格的 tkinter 桌面程序。 ttkbootstrap …...

基于RK3568的Linux驱动开发—— GPIO知识点(二)
authordaisy.skye的博客_CSDN博客-嵌入式,Qt,Linux领域博主系列基于RK3568的Linux驱动开发——GPIO知识点(一)_daisy.skye的博客-CSDN博客 查看goio使用情况 cat /sys/kernel/debug/gpio 1|rk3568_r:# cat /sys/kernel/debug/gpio gpiochip0: GPIOs 0-3…...

item_get-获得aliexpress商品详情API的调用参数说明
item_get-获得aliexpress商品详情 aliexpress.item_get 公共参数 名称类型必须描述keyString是调用key(免)(测)(试)secretString是调用密钥api_nameString是API接口名称(包括在请求地址中&…...

【Python_Scrapy学习笔记(三)】Scrapy框架之全局配置文件settings.py详解
Scrapy框架之全局配置文件settings.py详解 前言 settings.py 文件是 Scrapy框架下,用来进行全局配置的设置文件,可以进行 User-Agent 、请求头、最大并发数等的设置,本文中介绍 settings.py 文件下的一些常用配置 正文 1、爬虫的项目目录…...

spark读写时序数据库 TDengine 错误总结
最近在用spark读取、写入TDengine 数据库遇到了这样一个问题: JDBCDriver找不到动态链接库(no taos in java.library.path) 我本地都好好的,但是一上服务器写入就会报这个错误,看了很久没有排查出问题,后…...

Web中间件常见漏洞
一、IIS中间组件 1、PUT漏洞 原理:IIS开启了WebDAV,配置了可以写入的权限,造成了任意文件上传漏洞。 防御:关闭webDAV;关闭写入权限 2、短文件名猜解 原理: IIS的短文件名机制,可以暴力破解…...

Python Web 深度学习实用指南:第三部分
原文:Hands-On Python Deep Learning for the Web 协议:CC BY-NC-SA 4.0 译者:飞龙 本文来自【ApacheCN 深度学习 译文集】,采用译后编辑(MTPE)流程来尽可能提升效率。 不要担心自己的形象,只关…...

C#基础学习--预处理指令
目录 什么是预处理指令 基本规则 #define 和 #undef 指令 条件编译 条件编译结构 诊断指令 行号指令 编辑 区域指令 #pragam warning 指令 什么是预处理指令 源代码指定了程序的定义,预处理指令指示编译器如何处理源代码 基本规则 #define 和 #undef 指令…...

Spring Boot 接口加解密
1. 介绍 在我们日常的Java开发中,免不了和其他系统的业务交互,或者微服务之间的接口调用 如果我们想保证数据传输的安全,对接口出参加密,入参解密。 但是不想写重复代码,我们可以提供一个通用starter,提…...

大公司为什么禁止SpringBoot项目使用Tomcat?
前言 在SpringBoot框架中,我们使用最多的是Tomcat,这是SpringBoot默认的容器技术,而且是内嵌式的Tomcat。同时,SpringBoot也支持Undertow容器,我们可以很方便的用Undertow替换Tomcat,而Undertow的性能和内…...

2023年第十三届MathorCup高校数学建模挑战赛|A题|量子计算机在信用评分卡组合优化中的应用
目录 题目详情 最终收入 贷款利息收入 - 坏账损失 赛题说明 1:流程简化及示例 赛题说明 2:QUBO 模型简介 赛题说明 3:赛题数据 问题 题目详情 在银行信用卡或相关的贷款等业务中,对客户授信之前,需…...

linux下搭建Hbase分布式数据库
文章目录 Hbase概念1.安装Hbase1.jdk的配置2.安装hbase 2.启动和操作1.启动服务2 **web-ui访问地址:http://node01:16010/master-status** 3.简单的操作1.连接 HBase2.帮助命令3.创建一张表 create a table4.使用查看表是否存在5.describe 查看表描述6.put命令插入数据到表7. s…...

unity,射手游戏
文章目录 介绍一,制作玩家具体函数脚本PlayerCharacter三、 制作玩家控制脚本 PlayerController,调用上面的函数方法四、 制作子弹脚本 shell五、 给玩家挂载脚本六、 制作坦克脚本七、 给坦克添加组件八、 开始游戏,播放动画九、 下载 介绍 …...

摒弃单一变现手段,开拓多元商业模式,破解场景单一APP盈利难题!
工具类APP已成为人们生活、工作中不可或缺的一部分,包括天气服务、搜索、日历等细分领域,在用户中存在巨大的市场需求。但是,这类APP也面临着一些难以避免的问题。 比如功能单一、用户停留时间较短、可替代性强等,这些问题会影响…...

JavaEE-轻松了解网络原理之TCP协议
目录 TCP协议TCP协议数据格式TCP原理确认应答超时重传连接管理三次握手四次挥手 滑动窗口流量控制拥塞控制延迟应答捎带应答面向字节流异常问题 TCP协议 TCP,即Transmission Control Protocol,传输控制协议. TCP协议数据格式 16位源端口号与16位目的端…...

薪资17K是一个怎样的水平?来看看98年测试工程师的面试全过程…
我的情况 大概介绍一下个人情况,男,本科,三年多测试工作经验,懂python,会写脚本,会selenium,会性能,然而到今天都没有收到一份offer!从年后就开始准备简历,年…...

OpenCV3 和 Qt5 计算机视觉:11~12
原文:Computer Vision with OpenCV 3 and Qt5 协议:CC BY-NC-SA 4.0 译者:飞龙 本文来自【ApacheCN 计算机视觉 译文集】,采用译后编辑(MTPE)流程来尽可能提升效率。 当别人说你没有底线的时候,…...

R包编写流程
文章目录 所需工具Step 1: 创建R项目Step 2: 在R文件夹中添加函数Step 3: 编辑元数据Step 4: 文档化Step 5: 检查包Step 6: 打包重要参考: 所需工具 R包的编写需要的工具包有:devtools,Rtools Step 1: 创建R项目 devtools::create_package…...

试验GPT写文章书
试验GPT写文章书 写一本名叫《寻找人生目标的十种方法》 回答 2023/4/22 16:12:31 很高兴为您提供以下内容,这是一本关于寻找人生目标的十种方法的建议和思考。 《寻找人生目标的十种方法》 第一章:明确自己的价值观 了解自己内心真正想要追求的东…...

class与typename的异同
一、class与typename的相同点 typename关键字常用于函数模板,这里首先引入函数模板的概念:函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定 类型版本 //函数模板格式…...