unity学习24:场景scene相关生成,加载,卸载,加载进度,异步加载场景等

目录

1 场景数量 SceneManager.sceneCount

2 直接代码生成新场景 SceneManager.CreateScene

3 场景的加载

3.1 用代码加载场景,仍然build setting里先加入配置

3.2 卸载场景 SceneManager.UnloadSceneAsync();

3.3 同步加载场景 SceneManager.LoadScene

3.3.1  两种加载方式

3.4 异步加载场景

3.5 测试代码

3.5.1 有问题的测试代码

(代码创建的新Scene 需要手动去build Setting添加?)

3.6 场景的叠加效果

4 同步和异步

4.1  同步和异步

4.2 多线程  协程

4.2.1 多线程

4.2.2 协程

4.3 异步加载场景UnityEngine.SceneManagement;

4.3.1 除了默认的,还需要额外导入其他包

4.3.2  测试异步跳转场景,可成功

4.3.3 详细代码和注释

4.3.4 注释内容

5 加载进度

5.1  对应的协程的进度  operation1.progress

5.2  关于进度的数值

5.3 对应代码

5.4 如何做个显示的UI进度条

6 按条件跳转新地图

6.1 延迟跳转

6.2 测试代码


1 场景数量 SceneManager.sceneCount

  • //统计已经加载的场景数量
  • Debug.Log(SceneManager.sceneCount);

2 直接代码生成新场景 SceneManager.CreateScene

  • // 代码里可以创建新场景:直接用代码
  • Scene scene3=SceneManager.CreateScene("Scene3");

3 场景的加载

3.1 用代码加载场景,仍然build setting里先加入配置

  • 也是要注意,新场景的加载模式
  • 用代码创建的Scene也可以现在build setting里先加入配置

Scene 'Scene3' couldn't be loaded because it has not been added to the build settings or the AssetBundle has not been loaded.
To add a scene to the build settings use the menu File->Build Settings...
UnityEngine.SceneManagement.SceneManager:LoadScene (string)
SceneTest:Start () (at Assets/SceneTest.cs:43)

3.2 卸载场景 SceneManager.UnloadSceneAsync();

  • 卸载场景
  • SceneManager.UnloadSceneAsync("Scene3");

3.3 同步加载场景 SceneManager.LoadScene

  • //同步加载场景,卡顿,等待
  • SceneManager.LoadScene("Scene3");
  • SceneManager.loadScene("scene3",LoadSceneMode.Single);
  • SceneManager.loadScene("scene3",LoadSceneMode.Additive);

3.3.1  两种加载方式

只加载1个,替换之前的Scene

  • SceneManager.LoadScene("Scene2")  
  • 默认方式是 LoadSceneMode.Single
  • SceneManager.LoadScene("Scene2",LoadSceneMode.Single)  

新的场景加载,老的也在,相当于同时都加载生效

  • SceneManager.LoadScene("Scene2",LoadSceneMode.Additive) 

3.4 异步加载场景

  • //异步加载场景
  • //SceneManager.loadSceneAsync("scene3");

3.5 测试代码

3.5.1 有问题的测试代码

(代码创建的新Scene 需要手动去build Setting添加?)

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.SceneManagement;

public class SceneTest : MonoBehaviour
{
    // Start is called before the first frame update
    void Start()
    {
        ///先查看当前Scene
        //获取当前场景
        Scene scene1=SceneManager.GetActiveScene();

        //场景名称
        Debug.Log(scene1.name);

         //场景路径
        Debug.Log(scene1.path);      
    
        //场景索引
        Debug.Log(scene1.buildIndex);   
        GameObject[] gb1=scene1.GetRootGameObjects();
        Debug.Log(gb1.Length);

        //跳转场景
        //SceneManager.LoadScene(2);
        //SceneManager.LoadScene("Scene2");

        //调用异步的Start1
        Start1();

        //统计已经加载的场景数量
        Debug.Log(SceneManager.sceneCount);

        //创建新场景:直接用代码
        Scene scene3=SceneManager.CreateScene("Scene3");

        //卸载场景
        //SceneManager.UnloadSceneAsync("Scene3");

        //同步加载场景,卡顿,等待
        SceneManager.LoadScene("Scene3");
        //SceneManager.loadScene("scene3",LoadSceneMode.Single);
        //SceneManager.loadScene("scene3",LoadSceneMode.Additive);

        //异步加载场景
        //SceneManager.loadSceneAsync("scene3");


    }

        async void Start1()
        {
            AsyncOperation asyncLoad = SceneManager.LoadSceneAsync(2);
            // 等待场景加载完成
            while (!asyncLoad.isDone)
            {
                await System.Threading.Tasks.Task.Yield();
            }
            // 场景加载完成后获取信息
            Debug.Log(SceneManager.GetActiveScene().name);

            //获取当前场景
            //新定义1个scene2  Scene scene2=
            Scene scene2=SceneManager.GetActiveScene();

            //场景是否已经加载, 但是可能还没有激活新的Scene
            Debug.Log(scene2.isLoaded); 

            ///再次查看当前Scene
            //场景名称
            Debug.Log(scene2.name);

            //场景路径
            Debug.Log(scene2.path);      
        
            //场景索引
            Debug.Log(scene2.buildIndex);   
            GameObject[] gb2=scene2.GetRootGameObjects();
            Debug.Log(gb2.Length);
        }


    // Update is called once per frame
    void Update()
    {
        
    }
}

3.6 场景的叠加效果

  • 多个场景一起生效,叠加,效果
  • 从 Hierarchy 窗口里可以看到
  • 现在有3个场景,同时生效了,场景内的gb也都显示出来了

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.SceneManagement;

public class SceneTest : MonoBehaviour
{
    // Start is called before the first frame update
    void Start()
    {
        ///先查看当前Scene
        //获取当前场景
        Scene scene1=SceneManager.GetActiveScene();

        //场景名称
        Debug.Log(scene1.name);

         //场景路径
        Debug.Log(scene1.path);      
    
        //场景索引
        Debug.Log(scene1.buildIndex);   
        GameObject[] gb1=scene1.GetRootGameObjects();
        Debug.Log(gb1.Length);

        //跳转场景
        //SceneManager.LoadScene(2);
        //SceneManager.LoadScene("Scene2");

        //调用异步的Start1
        Start1();

        //统计已经加载的场景数量
        Debug.Log(SceneManager.sceneCount);

        //创建新场景:直接用代码
        Scene scene3=SceneManager.CreateScene("Scene3");

        //卸载场景
        //SceneManager.UnloadSceneAsync("Scene3");

        //同步加载场景,卡顿,等待
        SceneManager.LoadScene("Scene3");
        //SceneManager.loadScene("scene3",LoadSceneMode.Single);
        //SceneManager.loadScene("scene3",LoadSceneMode.Additive);

        //异步加载场景
        //SceneManager.loadSceneAsync("scene3");


    }

        async void Start1()
        {
            AsyncOperation asyncLoad = SceneManager.LoadSceneAsync(2,LoadSceneMode.Additive);
            // 等待场景加载完成
            while (!asyncLoad.isDone)
            {
                await System.Threading.Tasks.Task.Yield();
            }
            // 场景加载完成后获取信息
            Debug.Log(SceneManager.GetActiveScene().name);

            //获取当前场景
            //新定义1个scene2  Scene scene2=
            Scene scene2=SceneManager.GetActiveScene();

            //场景是否已经加载, 但是可能还没有激活新的Scene
            Debug.Log(scene2.isLoaded); 

            ///再次查看当前Scene
            //场景名称
            Debug.Log(scene2.name);

            //场景路径
            Debug.Log(scene2.path);      
        
            //场景索引
            Debug.Log(scene2.buildIndex);   
            GameObject[] gb2=scene2.GetRootGameObjects();
            Debug.Log(gb2.Length);
        }


    // Update is called once per frame
    void Update()
    {
        
    }
}

4 同步和异步

4.1  同步和异步

  • 同步: 纯线性
  • 异步: 也是线性,但是只是逻辑上还是线性。但是有了分支,把消耗时间的操作,放到其他线程里去执行

下面几个图都是网上找的参考的

4.2 多线程  协程

  • 多线程
  • 协程

4.2.1 多线程

多线程是一种同时运行多个执行路径的技术,每个执行路径称为一个线程。

多个线程可以在多核CPU上真正并行运行,或者在单核CPU上通过时间片轮转模拟并发。多线程通过操作系统调度,能够充分利用计算资源,在处理I/O密集型和CPU密集型任务时具有优势。

特点:
每个线程都有独立的栈空间和执行路径。
线程之间可以共享内存数据,因此需要进行同步控制,以避免数据竞争和死锁问题。
线程调度由操作系统控制,可能涉及上下文切换,带来一定的开销。

4.2.2 协程

协程是一种比线程更轻量级的并发实现方式。

与多线程不同,协程不是由操作系统调度,而是由编程语言或运行时环境来管理。

协程可以在需要时暂停自身,并将控制权交还给调用方,稍后再恢复执行。

它们适用于处理需要频繁暂停和恢复的任务,如异步I/O操作。

特点:
协程不会并行运行,单个线程中可以运行多个协程。
协程之间共享执行线程,但不需要上下文切换,切换开销非常小。
协程适用于I/O密集型任务,如网络请求、文件读写等,能够实现高效的异步操作。

4.3 异步加载场景UnityEngine.SceneManagement;

4.3.1 除了默认的,还需要额外导入其他包

using UnityEngine;

using UnityEngine.SceneManagement;

4.3.2  测试异步跳转场景,可成功

 去掉各种注释的代码如下

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
//导入场景管理类
using UnityEngine.SceneManagement;

public class AsyncTest : MonoBehaviour
{

    void Start()
    {
        StartCoroutine(loadScene());
        
    }

    IEnumerator loadScene()
    {
        // 异步的协程
        AsyncOperation operation1=SceneManager.LoadSceneAsync(2);
        yield return operation1;
    }


    // Update is called once per frame
    void Update()
    {
        
    }
}

4.3.3 详细代码和注释

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
//导入场景管理类
using UnityEngine.SceneManagement;

public class AsyncTest : MonoBehaviour
{
    //异步,需要先声明SceneManager.LoadSceneAsync(2)的返回值
    //AsyncOperation operation1;

    // Start is called before the first frame update
    void Start()
    {
        //协程不能直接LoadScene(),必须新建一个协程方法StartCoroutine
        StartCoroutine(loadScene());
        
    }

    //以协程的方法来异步加载场景
    //必须单独写一个 loadScene()方法,而不能用Application.LoadScene()方法
    //这个异步的方法,有返回值,返回值是固定的IEnumerator
    IEnumerator loadScene()
    {
        // 异步的协程
        // SceneManager.LoadSceneAsync()有返回值,返回值类型AsyncOperation
        AsyncOperation operation1=SceneManager.LoadSceneAsync(2);
        yield return operation1;
    }




    // Update is called once per frame
    void Update()
    {
        
    }
}

4.3.4 注释内容

     //异步,需要先声明SceneManager.LoadSceneAsync(2)的返回值

    // 也是可以在函数内,使用 operation1 时当时声明,  但是函数内声明的,函数外就无法条用operation1,所以为了外面可以调用,还是在外面声明

    //AsyncOperation operation1;

    // Start is called before the first frame update

    void Start()

    {

        //协程不能直接LoadScene(),必须新建一个协程方法StartCoroutine

        StartCoroutine(loadScene());

       

    }

    //以协程的方法来异步加载场景

    //必须单独写一个 loadScene()方法,而不能用Application.LoadScene()方法

    //这个异步的方法,有返回值,返回值是固定的IEnumerator

    IEnumerator loadScene()

    {

        // 异步的协程

        // SceneManager.LoadSceneAsync()有返回值,返回值类型AsyncOperation

       // 也可以在函数外,最开始声明

        AsyncOperation operation1=SceneManager.LoadSceneAsync(2);

        yield return operation1;

    }

5 加载进度

5.1  对应的协程的进度  operation1.progress

  • operation1.progress
  • 适合放在 void Update() 方法里去实现,因为逐帧加载。

5.2  关于进度的数值

  • unity里,输出进度 0-0.9,其中0.9 在unity里就是100%
  • 实际游戏里
  • 有的0.9-100%飞速,有的是按比例0.9折算到100%
  • 场景小异步加载也就很快

5.3 对应代码

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
//导入场景管理类
using UnityEngine.SceneManagement;

public class AsyncTest : MonoBehaviour
{
    //异步,需要先声明SceneManager.LoadSceneAsync(2)的返回值
    AsyncOperation operation1;

    // Start is called before the first frame update
    void Start()
    {
        //协程不能直接LoadScene(),必须新建一个协程方法StartCoroutine
        StartCoroutine(loadScene());
        
    }

    //以协程的方法来异步加载场景
    //必须单独写一个 loadScene()方法,而不能用Application.LoadScene()方法
    //这个异步的方法,有返回值,返回值是固定的IEnumerator
    IEnumerator loadScene()
    {
        // 异步的协程
        // SceneManager.LoadSceneAsync()有返回值,返回值类型AsyncOperation
        operation1=SceneManager.LoadSceneAsync(2);
        yield return operation1;
    }

    // Update is called once per frame
    void Update()
    {
        Debug.Log(operation1.progress);
    }
}

5.4 如何做个显示的UI进度条

。。。。是个问题,学到了再说

6 按条件跳转新地图

6.1 延迟跳转

  • 跳转地图也可以不是即可生效的
  • 可以加定时器, 按任意键跳转,下面的例子可以实现10秒后跳转
  • 除了延迟跳转新地图,也可以设置其他条件:比如 响应键盘,UI等等

6.2 测试代码

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
//导入场景管理类
using UnityEngine.SceneManagement;

public class AsyncTest : MonoBehaviour
{
    //异步,需要先声明SceneManager.LoadSceneAsync(2)的返回值
    AsyncOperation operation1;
    float timer1=0;

    // Start is called before the first frame update
    void Start()
    {
        //协程不能直接LoadScene(),必须新建一个协程方法StartCoroutine
        StartCoroutine(loadScene());
        
    }

    //以协程的方法来异步加载场景
    //必须单独写一个 loadScene()方法,而不能用Application.LoadScene()方法
    //这个异步的方法,有返回值,返回值是固定的IEnumerator
    IEnumerator loadScene()
    {
        // 异步的协程
        // SceneManager.LoadSceneAsync()有返回值,返回值类型AsyncOperation
        operation1=SceneManager.LoadSceneAsync(2);
        operation1.allowSceneActivation=false;
        yield return operation1;
    }

    // Update is called once per frame
    void Update()
    {
        Debug.Log(operation1.progress);
        timer1=timer1+Time.deltaTime;
        if (timer1>10)
        {
        operation1.allowSceneActivation=true;
        }
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/962583.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

在线知识库创建与维护提升企业效率与知识共享能力

内容概要 在当今数字化快速发展的背景下,在线知识库逐渐成为企业管理信息的重要工具。其核心在于将知识进行系统化、结构化的整理和存储,便于员工获取和分享。这不仅提高了信息的访问效率,还促进了团队之间的协作。在线知识库的建立可以有效…

【Numpy核心编程攻略:Python数据处理、分析详解与科学计算】1.24 随机宇宙:生成现实世界数据的艺术

1.24 随机宇宙:生成现实世界数据的艺术 目录 #mermaid-svg-vN1An9qZ6t4JUcGa {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-vN1An9qZ6t4JUcGa .error-icon{fill:#552222;}#mermaid-svg-vN1An9qZ6t4JUc…

DeepSeek-R1 论文解读 —— 强化学习大语言模型新时代来临?

近年来,人工智能(AI)领域发展迅猛,大语言模型(LLMs)为通用人工智能(AGI)的发展开辟了道路。OpenAI 的 o1 模型表现非凡,它引入的创新性推理时缩放技术显著提升了推理能力…

进阶数据结构——高精度运算

目录 前言一、高精度运算的定义与背景二、高精度运算的实现方式三、高精度运算的算法实现四、高精度运算的应用场景五、代码模版(c)六、经典例题1.[高精度加法](https://www.lanqiao.cn/problems/1516/learning/?page1&first_category_id1&name…

MYSQL--一条SQL执行的流程,分析MYSQL的架构

文章目录 第一步建立连接第二部解析 SQL第三步执行 sql预处理优化阶段执行阶段索引下推 执行一条select 语句中间会发生什么? 这个是对 mysql 架构的深入理解。 select * from product where id 1;对于mysql的架构分层: mysql 架构分成了 Server 层和存储引擎层&a…

【4Day创客实践入门教程】Day2 探秘微控制器——单片机与MicroPython初步

Day2 探秘微控制器——单片机与MicroPython初步 目录 Day2 探秘微控制器——单片机与MicroPython初步MicroPython语言基础开始基础语法注释与输出变量模块与函数 单片机基础后记 Day0 创想启程——课程与项目预览Day1 工具箱构建——开发环境的构建Day2 探秘微控制器——单片机…

动态规划每日一练(四)

一、day1——最长数对链 题目链接&#xff1a; 646. 最长数对链 - 力扣&#xff08;LeetCode&#xff09;646. 最长数对链 - 给你一个由 n 个数对组成的数对数组 pairs &#xff0c;其中 pairs[i] [lefti, righti] 且 lefti < righti 。现在&#xff0c;我们定义一种 跟随…

事务02之锁机制

锁机制 文章目录 锁机制一&#xff1a;MySQL锁的由来与分类1&#xff1a;锁机制的分类 二&#xff1a;共享锁与排他锁1&#xff1a;共享锁(S锁)2&#xff1a;排他锁(X锁)3&#xff1a;锁的释放 二&#xff1a;表级别锁1&#xff1a;元数据锁(了解)2&#xff1a;意向锁3&#xf…

网络工程师 (8)存储管理

一、页式存储基本原理 &#xff08;一&#xff09;内存划分 页式存储首先将内存物理空间划分成大小相等的存储块&#xff0c;这些块通常被称为“页帧”或“物理页”。每个页帧的大小是固定的&#xff0c;例如常见的页帧大小有4KB、8KB等&#xff0c;这个大小由操作系统决定。同…

[EAI-026] DeepSeek-VL2 技术报告解读

Paper Card 论文标题&#xff1a;DeepSeek-VL2: Mixture-of-Experts Vision-Language Models for Advanced Multimodal Understanding 论文作者&#xff1a;Zhiyu Wu, Xiaokang Chen, Zizheng Pan, Xingchao Liu, Wen Liu, Damai Dai, Huazuo Gao, Yiyang Ma, Chengyue Wu, Bin…

(动态规划路径基础 最小路径和)leetcode 64

视频教程 1.初始化dp数组&#xff0c;初始化边界 2、从[1行到n-1行][1列到m-1列]依次赋值 #include<vector> #include<algorithm> #include <iostream>using namespace std; int main() {vector<vector<int>> grid { {1,3,1},{1,5,1},{4,2,1}…

cf1000(div.2)

Minimal Coprime最小公倍数 输入&#xff1a; 6 1 2 1 10 49 49 69 420 1 1 9982 44353 输出&#xff1a; 1 9 0 351 1 34371 代码

【单细胞第二节:单细胞示例数据分析-GSE218208】

GSE218208 1.创建Seurat对象 #untar(“GSE218208_RAW.tar”) rm(list ls()) a data.table::fread("GSM6736629_10x-PBMC-1_ds0.1974_CountMatrix.tsv.gz",data.table F) a[1:4,1:4] library(tidyverse) a$alias:gene str_split(a$alias:gene,":",si…

事务04之死锁,锁底层和隔离机制原理

死锁和事务底层原理 文章目录 死锁和事务底层原理一&#xff1a;MySQL中的死锁现象1&#xff1a;何为死锁1.1&#xff1a;死锁的概念1.2&#xff1a;死锁产生的四个必要条件&#xff1a; 2&#xff1a;MySQL的死锁2.1&#xff1a;死锁的触发2.2&#xff1a;MySQL的死锁如何解决…

Maven的单元测试

1. 单元测试的基本概念 单元测试&#xff08;Unit Testing&#xff09; 是一种软件测试方法&#xff0c;专注于测试程序中的最小可测试单元——通常是单个类或方法。通过单元测试&#xff0c;可以确保每个模块按预期工作&#xff0c;从而提高代码的质量和可靠性。 2.安装和配…

VirtualBox:跨磁盘导入已存的vdi磁盘文件顺便测试冷迁移

目录 1.背景 2.目的 3.步骤 3.1 安装在移动硬盘上 3.2.接管现有主机磁盘上的虚拟机 3.3接管迁移到移动硬盘的虚拟机 4. 结论 1.背景 电脑重新做了系统&#xff0c;然后找不到virtualbox的启动程序了&#xff0c;另外电脑磁盘由于存储了其他文件已经爆红&#xff0c;无法…

二级C语言:二维数组每行最大值与首元素交换、删除结构体的重复项、取出单词首字母

目录 一、程序填空 --- 二维数组每行最大值与首元素交换 题目 分析 知识点 --- 交换语句 二、程序修改 --- 删除结构体的重复项 题目 分析 三、程序设计 --- 取出单词首字母 题目 分析 前言 本章讲解&#xff1a;二维数组每行最大值与首元素交换、删除结构体的重复项…

优盘恢复原始容量工具

买到一个优盘&#xff0c;显示32mb&#xff0c;我见过扩容盘&#xff0c;但是这次见到的是缩容盘&#xff0c;把2g的容量缩成32MB了&#xff0c;首次见到。。用芯片查询工具显示如下 ChipsBank(芯邦) CBM2199E 使用以下工具&#xff0c;恢复原始容量。。 其他CMB工具可能不行…

面对企业文件交换难题,镭速跨网文件交换系统是如何解决的?

在当今这个数字化快速发展的时代&#xff0c;企业越来越依赖于数据交换来维持其业务运作。无论是内部网络之间的沟通还是与外部合作伙伴的数据共享&#xff0c;高效且安全的跨网文件交换都显得尤为重要。然而&#xff0c;在实际操作中&#xff0c;许多企业面临着各种各样的挑战…

黑马点评 - 商铺类型缓存练习题(Redis List实现)

首先明确返回值是一个 List<ShopType> 类型那么我们修改此函数并在 TypeService 中声明 queryTypeList 方法&#xff0c;并在其实现类中实现此方法 GetMapping("list")public Result queryTypeList() {return typeService.queryTypeList();}实现此方法首先需要…