[Unity3D]异步加载游戏场景与异步加载游戏资源进度条

简介: 摘要: 异步任务相信大家应该不会陌生,那么本章内容MOMO将带领大家学习Unity中的一些异步任务。在同步加载游戏场景的时候通常会使用方法 Application.LoadLevel(“yourScene”); 这句代码执行完毕后程序会干什么呢??如下 ...     异步任务相信大家应该不会陌生,那么下面介绍一下Unity中的一些异步任务。
摘要: 异步任务相信大家应该不会陌生,那么本章内容MOMO将带领大家学习Unity中的一些异步任务。在同步加载游戏场景的时候通常会使用方法 Application.LoadLevel(“yourScene”); 这句代码执行完毕后程序会干什么呢??如下 ...

    异步任务相信大家应该不会陌生,那么下面介绍一下Unity中的一些异步任务。在同步加载游戏场景的时候通常会使用方法 Application.LoadLevel(“yourScene”);  这句代码执行完毕后程序会干什么呢??如下图所示,这是我随便找了一个游戏场景, 在Hierarchy视图中我们可以看到该场景中“天生”的所有游戏对象。天生的意思就是运行程序前该场景中就已经存在的所有游戏对象。然后这些对象就会在执行完Application.LoadLevel(“yourScene”);方法后加载至内存当中。如果该场景中的游戏对象过多那么瞬间将会出现卡一下的情况,因为LoadLevel()方法是同步进行的。MOMO把这种加载起个名字叫A形式加载。

 

 

下面我说说“后天“加载的游戏对象。意思是这些游戏对象是通过脚本动态的创建出来的。比如常用方法 :

[代码]

GameObject Obj = (GameObject)Instantiate(prefab);

这句代码执行完毕后同样会在Hierarchy视图中添加对应的游戏对象。MOMO把这种加载起个名字叫B形式加载。

 

下面我们学习异步加载游戏场景,异步异步顾名思义就是不影响当前游戏场景的前提下加载新场景。通常异步加载的方式分为两种:第一种是异步加载新游戏场景,当新场景加载完成后进入新场景并且销毁之前的场景。第二种:同样异步加载新场景,新场景加载完毕后,保留旧场景的游戏对象并且进入新场景。 这里加载的内容就是上面提到的A形式加载。然后B形式加载不会记入这里的加载。

第一种异步加载游戏场景对应的方法是:

[代码]:

Application.LoadLevelAsync("yourScene");

这两种方法加载的方式完全一样。异步加载其实重要还是应用于游戏LOADING界面,毕竟LOADING如果采用同步的机制会影响用户体验,说到这里MOMO告诉大家如何在Unity中制作游戏进度条。我们应当在Unity中创建一个专门用于读取进度的场景,假设A场景到C场景,我们应当让A场景先到读取进度的场景B场景,当异步任务完成后在进入C场景。 A – 》B -》 C ,在B场景中绘制游戏进度条或读取动画因为B场景仅仅是个显示LOADING动画的场景,所以读取该场景是瞬间就完成的。

程序在切换场景时应当有一个全全局的静态变量来记录简要读取的场景名称。这里简单的写一下。

[代码]:

using UnityEngine;
using System.Collections;

public class Globe
{
         //在这里记录当前切换场景的名称
	 public static string loadName;
}
在A场景中通过某些触发条件 调用LoadLevel进入B场景。

[代码]:

//记录LOADING场景中需要读取的C场景名称
Globe.loadName = "C";
//先进入B场景
Application.LoadLevel ("B");
OK我们在B场景中异步读取C场景与 播放读取动画,Loading.cs 绑定在B场景的摄像机对象身上。当C场景异步读取完毕后即可直接进入C场景。

[代码]:

using UnityEngine;
using System.Collections;

public class Loading : MonoBehaviour {

	private float fps = 10.0f;
	private float time;
	//一组动画的贴图,在编辑器中赋值。
	public Texture2D[] animations;
	private int nowFram;
	//异步对象
	AsyncOperation async;

	//读取场景的进度,它的取值范围在0 - 1 之间。
	int progress = 0;

	void Start()
	{
		//在这里开启一个异步任务,
		//进入loadScene方法。
	  	StartCoroutine(loadScene());
	}

	//注意这里返回值一定是 IEnumerator
	IEnumerator loadScene()
	{
		//异步读取场景。
		//Globe.loadName 就是A场景中需要读取的C场景名称。
		async = Application.LoadLevelAsync(Globe.loadName);

		//读取完毕后返回, 系统会自动进入C场景
		yield return async;

	}

	void OnGUI()
	{
		//因为在异步读取场景,
		//所以这里我们可以刷新UI
		DrawAnimation(animations);

	}

	void Update()
	{

		//在这里计算读取的进度,
		//progress 的取值范围在0.1 - 1之间, 但是它不会等于1
		//也就是说progress可能是0.9的时候就直接进入新场景了
		//所以在写进度条的时候需要注意一下。
		//为了计算百分比 所以直接乘以100即可
		progress =  (int)(async.progress *100);

		//有了读取进度的数值,大家可以自行制作进度条啦。
		Debug.Log("xuanyusong" +progress);

	}
	//这是一个简单绘制2D动画的方法,没什么好说的。
	void   DrawAnimation(Texture2D[] tex)
	{

		time += Time.deltaTime;

		 if(time >= 1.0 / fps){

      	 	 nowFram++;

      	 	 time = 0;

      	 	 if(nowFram >= tex.Length)
      	 	 {
      	 	 	nowFram = 0;
      	 	 }
        }
		GUI.DrawTexture(new Rect( 100,100,40,60) ,tex[nowFram] );

        //在这里显示读取的进度。
		GUI.Label(new Rect( 100,180,300,60), "lOADING!!!!!" + progress);

	}

}

OK 下面我们继续学习在游戏场景中加载对象,文章的开始MOMO已经告诉大家,游戏场景中Hierarchy视图中的所有的对象在切换场景的时候都会加载。其实有一种方法可以让某些游戏对象不会被加载,如下图所示,首先在Hierarchy视图中选择一个游戏对象,在右侧监测面板视图中我们可以看到一个 “小对勾”默认情况下是勾选状态,说明该游戏对象处于激活状态,如果点掉的话该对象将被隐藏。这个小功能在开发中其实用处非常大,请大家务必记住哈。

 

 

此时此刻大家相像一个游戏场景,默认进入的时候是没有任何游戏对象的,然后运行游戏时开启一个异步任务将它们一个一个的加载显示出来,这种方式适合异步的加载一个比较大的游戏场景。

Test.cs 把它挂在摄像机对象中。

[代码]:

using UnityEngine;
using System.Collections;

public class Test : MonoBehaviour {

	//这里是需要加载激活的游戏对象
	public GameObject  [] Objects;

	//当前加载的进度
	int load_index =0;
	void Start ()
	{
	    //开启一个异步任务,加载模型。
		StartCoroutine(loadObject());
	}

	IEnumerator loadObject()
	{
		//便利所有游戏对象
		foreach(GameObject obj in Objects)
		{
			//激活游戏对象
			obj.active = true;
			//记录当前加载的对象
			load_index ++;

			//这里可以理解为通知主线程刷新UI
			yield return 0;
		}
		//全部便利完毕返回
		yield return 0;
	}

	void OnGUI ()
	{
	    //显示加载的进度
		GUILayout.Box("当前加载的对象ID是: " + load_index);
	}
}

如下图所示,我们把需要加载的游戏对象以数组的形式放在Objects数组中,因为这些对象属于未激活状态,所以不能通过Find 等方法在脚步那种中找到他们。讲到这里我们在说说 编辑器赋值与代码中赋值的区别,编辑器中赋值所消耗的时间都会记在loadlevel ()读取场景中。而代码中使用Resource.load()这类方法所消耗的时间会记在脚本中。开发中还得自行的把握一下把loading加在那里。

 

 

当然我们还可以使用Instantiate(prefab);方法来动态的创建游戏对象。

Main.cs 把它挂在摄像机中。

[代码]:

using UnityEngine;
using System.Collections;

public class Main : MonoBehaviour
 {

	public int count;
	//在编辑器中预设一个游戏对象
	public  GameObject prefab;

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

	void OnGUI()
	{
		GUILayout.Box("游戏对象已经加载到 : " + count);
	}

	IEnumerator loaditem()
	{
		//开始加载游戏对象
		for(int i =0; i< 1000; i++)
		{

			Instantiate(prefab);
			count = i;
			//可以理解为刷新UI,显示新加载的游戏对象
			yield return 0;
		}
		//结束
		yield return 0;
	}
}
运行游戏后该游戏对象会循环1000遍逐个创建,不影响主线程。那么今天我们其实学习最多的就是StartCoroutine(),其实就是开启一个异步线程,这里可能有朋友会问Thread可以代替它吗? 答案是不行, 比如查询数据库的时候如果用Thread的话Unity就会报错说不能在线程中查询,但是在StartCoroutine()中就可以完成,所以开发中大家可以尝试着使用它,我们还可以使用StopCoroutine(“name”)来关闭一个正在执行的异步线程。
相关文章
|
图形学
unity3d UGUI常用游戏进度条实现方式
测试.png 直接将脚本挂载到进度条image对象上即可,这种方式可以解决当进度条使用图片的时候,防止图片拉伸变形 using UnityEngine; using UnityEngine.
3013 0
|
3月前
|
JavaScript 前端开发 UED
Vue.js动画魔法:解锁流畅过渡,让每一次交互都成为用户心中的小确幸!
【8月更文挑战第30天】在Vue.js中,动画与过渡效果不仅是视觉点缀,更是提升用户体验的关键。通过流畅的动态效果,应用的互动性和吸引力得以增强,从而提高用户满意度和参与度。`&lt;transition&gt;`和`&lt;transition-group&gt;`组件结合CSS过渡,可轻松实现元素的进入、离开及列表变化动画。合理的性能优化,如使用硬件加速,能避免页面卡顿,确保动画既美观又高效。下面是一个简单的淡入淡出效果示例,展示了如何利用Vue.js实现平滑的动画过渡。总之,恰当的动画设计能显著提升应用的用户体验。
59 0
Vue.js动画魔法:解锁流畅过渡,让每一次交互都成为用户心中的小确幸!
|
小程序 JavaScript API
小程序图片渲染
小程序图片渲染
109 0
|
编解码 缓存 图形学
unity中的渲染优化技术
unity中的渲染优化技术
|
JSON 小程序 JavaScript
小程序触底加载与下拉刷新功能的设计与实现
在最近做小程序的时候有这么一个很常见的需求,加载一个信息列表,要求需要触底加载和下拉刷新,我突然想起来掘金小册界面和这个需求很相似,接下来我给大家介绍一下我的实现方案。
371 0
小程序触底加载与下拉刷新功能的设计与实现
|
存储 缓存 数据处理
二、屏幕卡顿 及 iOS中的渲染流程解析
屏幕卡顿是指图形图像的在显示时出现了撕裂(即图片错位显示)、掉帧(重复显示同一帧数据)等问题,导致用户能直观的从屏幕上看到的一种异常现象
493 0
二、屏幕卡顿 及 iOS中的渲染流程解析
|
前端开发 C# 图形学
【Unity使用UGUI实现王者荣耀UI界面(一)】加载页面(进度条)
【Unity使用UGUI实现王者荣耀UI界面(一)】加载页面(进度条)
738 0
【Unity使用UGUI实现王者荣耀UI界面(一)】加载页面(进度条)
|
前端开发 图形学 Python
【Unity使用UGUI实现王者荣耀UI界面(二)】加载页面-静音按钮和页面完善
【Unity使用UGUI实现王者荣耀UI界面(二)】加载页面-静音按钮和页面完善
392 0
【Unity使用UGUI实现王者荣耀UI界面(二)】加载页面-静音按钮和页面完善
|
JavaScript
送你一个封装的移动端横屏展示的JS库
我们经常在移动端横屏展示一些东西,所以今天我们来介绍下怎么在移动端实现横屏展示。