1.AssetBundle原理
AssetBundle又称AB包,是Unity针对资源提供的一种用于资源存储的压缩包。通过将资源分布在不同的AB包中可以最大程度地减少运行时的内存压力,可以动态地加载和卸载AB包,继而有选择地加载内容。
AssetBundle 是一个存档文件,包含可在运行时由 Unity 加载的特定于平台的非代码资源(比如模型、纹理、预制件、音频剪辑甚至整个场景)。
注意:AssetBundle资源管理包含资源对象,但是不包含类,无法存储C#脚本,所以对于脚本更新使用Lua。AB包不能重复进行加载,当AB包已经加载进内存后必须卸载后才能重新加载。
(1)AssetBundle的结构
AssetBundle由两部分组成:包头和数据段。
包头:包含有关AssetBundle的信息,例如其标识符、压缩类型和清单(manifest:manifest是一个用对象名字做key的查找表,每个条目都提供一个字节索引,指示在AssetBundle的数据段中可以找到给定对象的位置。后面再处理资源依赖关系时候会写到这部分内容),在大多数平台上,此查找表是作为平衡搜索树实现的。具体而言,Windows和OSX衍生平台(包括iOS)采用红黑树。因此,随着AssetBundle中资产数量的增加,构建清单所需的时间将线性增加。
数据段:数据段包含通过序列化AssetBundle中的资产而生成的原始数据。
三种压缩方式:
LZMA是流压缩方式(stream-based)。流压缩再处理整个数据块时使用同一个字典,它提供了最大可能的压缩率,但是只支持顺序读取。所以加载AB包时,需要将整个包解压,会造成卡顿和额外内存占用。
LZ4是块压缩方式(chunk-based)。块压缩的数据被分为大小相同的块,并被分别压缩。如果需要实时解压随机读取,块压缩是比较好的选择。LoadFromFile()和LoadFromStream()都只会加载AB包的Header,相对LoadFromMemory()来说大大节省了内存。 压缩稍大,解压快,用什么解压什么,内存占用低,建议使用。
不使用压缩,数据段将保留为原始字节流。不压缩,解压快,包较大,不建议使用。
(2)AssetBundle特点
- AB包存储位置自定义,继而可放入可读可写的路径下便于实现热更新
- AB包自定义压缩方式,可以选择不压缩或选择LZMA和LZ4等压缩方式,减小包的大小,更快的进行网络传输。
- 资源可分布在不同的AB包中,最大程度减少运行时的内存压力, 可做到即用即加载,有选择的加载需要的内容。
- AB包支持后期进行动态更新,显著减小初始安装包的大小,非核心资源以AB包形式上传服务器,后期运行时动态加载,提高用户体验。
- 多个资源分布在不同的AB包可能会出现一个预制体的贴图等部分资源不在同一个包下,直接加载会出现部分资源丢失的情况,即AB包之间是存在依赖关系的,在加载当前AB包时需要一并加载其所依赖的包。
- 打包完成后,会自动生成一个主包(主包名称随平台不同而不同),主包的manifest下会存储有版本号、校验码(CRC)、所有其它包的相关信息(名称、依赖关系)
2.AssetBundle打包
(1)设置模型资源AssetBundle的位置(先bundlename 赋值,再设置,,创建Editor文件夹,创建继承Editor脚本)
[MenuItem("Tools/SetAssetBundleName")]
static void SetResourcesAssetBundleName()
{
DirectoryInfo root = new DirectoryInfo(Application.dataPath + "/Resources/XXX");
SearchRoot(root, "");
AssetDatabase.RemoveUnusedAssetBundleNames();
AssetDatabase.Refresh();
}
static void SearchRoot(DirectoryInfo root,string foldeIndex)
{
foreach (FileInfo itemFile in root.GetFiles())
{
if (itemFile.Extension.Contains(".prefab"))
{
string path = itemFile.FullName.Replace('\\', '/').Substring(AssetBundleConfig.PROJECT_PATH.Length);
Debug.Log("path=" + path);
var importer = AssetImporter.GetAtPath(path);
if (importer)
{
//string name = path.Substring("Assets/Resources/PageItemModel/".Length);
string name = foldeIndex+"/"+itemFile.Name;
name = name.Substring(1, name.LastIndexOf('.')-1);
Debug.Log("name=" + (GetLower(name)+ AssetBundleConfig.SUFFIX));
importer.assetBundleName = GetLower(name) + AssetBundleConfig.SUFFIX;
}
}
}
foreach (DirectoryInfo subFolder in root.GetDirectories())
{
if (subFolder.Name.Equals("fbm") || subFolder.Name.Equals("Materials") || subFolder.Name.Equals("Texture"))
{
continue;
}
SearchRoot(subFolder, foldeIndex+"/"+subFolder.Name.Substring(2));
}
}
static string GetLower(string str)
{
char[] ch = str.ToCharArray();
if (str != null)
{
for (int i = 0; i < ch.Length; i++)
{
if (!isChinese(ch[i]))
{
if ((int)ch[i] > 64 && (int)ch[i] < 91)
{
ch[i] = System.Char.ToLower(ch[i]);
}
}
}
}
string lower = new string(ch);
return lower;
}
static bool isChinese(char c)
{
return c >= 0x4E00 && c <= 0x9FA5;
}
(2)打包功能
private static string OutPutPath = System.IO.Directory.GetCurrentDirectory() + "/0830";
[MenuItem("Tools/Build WebGl")]
static void BuildAllAssetBundles()
{
if (!Directory.Exists(OutPutPath))
{
Directory.CreateDirectory(OutPutPath);
}
BuildPipeline.BuildAssetBundles(OutPutPath, BuildAssetBundleOptions.ChunkBasedCompression, BuildTarget.WebGL);
AssetDatabase.Refresh();
}
//BuildTarget.WebGL 是WebGL平台,
打包后输出的文件在OutPutPath路径,放在你的服务器。
3.AssetBundle加载
(1)API
用于加载AssetBundle的API:
AssetBundle.LoadFromFile[Async]()
- 首选方法(在速度、磁盘使用和内存占用方面都很高效)
- 适用于从本地存储加载未压缩或LZ4压缩的AssetBundle
- 使用LZMA压缩的AssetBundle会被解压到内存中
- 使用LZ4压缩或未压缩的AssetBundle会直接从磁盘读取
AssetBundle.LoadFromStream[Async]()
- 适用于从流式数据中加载AssetBundle
- 使用LZMA压缩的AssetBundle会被解压到内存中
- 使用LZ4压缩或未压缩的AssetBundle会直接从流数据中读取
- 加载过程中不能Dispose流式对象
UnityWebRequest[AssetBundle].GetAssetBundle()
- 适用于从远程下载AssetBundle
- 能够缓存已下载的AssetBundle以便重用
- 缓存系统中的AssetBundle仅以文件名进行标识而不是使用URL
AssetBundle.LoadFromMemory[Async]()
- 不推荐使用
- 会产生AssetBundle的冗余副本
注意:LoadFromFile是从文件中加载AB包,它从一个给定的路径来加载AB包。如果AB包是LZ4加载方式,它只会加载AB包的Header,之后需要什么资源再加载那部分的AB包chunk。极大的减少了内存占用。(LoadFromFileAsync是它的异步版本)
LoadFromMemory是从内存中加载AB包,它从内存中的byte[]中加载AB包。它会完整的把AB包加载出来。(LoadFromMemoryAsync是它的异步版本)
LoadFromStream是从流中加载AB包,它从一个Stream中加载AB包。跟LoadFromFile一样,如果AB包是LZ4加载方式,它也是只会加载AB包的Header。(LoadFromStreamAsync是它的异步版本)
UnityWebRequest.GetAssetBundle是Unity中的跟网络相关的类,可以通过该类从网络中下载资源,之后加载成AB包。
用于从AssetBundle中加载Asset的API:
AssetBundle.LoadAsset[Async]()
- 首选方法
- 适用于加载AssetBundle中的单个Object
- 当要加载AssetBundle中超过66%的Object时,考虑使用LoadAllAssets()方法
AssetBundle.LoadAllAssets[Async]()
- 适用于一次性加载AssetBundle中的全部Object
- 比多次调用LoadAsset()更快
AssetBundle.LoadAssetWithSubAssets[Async]()
- 适用于从AssetBundle中加载含有多个嵌套Object的复合Asset
- 如果要加载的Object都来自于同一Asset并且他们和很多不相关的其他Object放在同一AssetBundle内,那么应该使用此方法
用于查询AssetBundle依赖的API
AssetBundleManifest.GetAllDependencies()
- 返回被AssetBundle所直接和间接依赖的所有AssetBundle的名称
AssetBundleManifest.GetDirectDependencies()
- 返回被AssetBundle所直接依赖的AssetBundle的名称
用于卸载AssetBundle的API
AssetBundle.Unload()
- 卸载此AssetBundle
- 如果参数unloadAllLoadedObjects传入false
- 此AssetBundle中的Asset的压缩数据文件将被卸载
- 无法再从此AssetBundle中加载任何Object
- 已经从此AssetBundle中加载的Object仍能正常工作
- 如果参数unloadAllLoadedObjects传入true
- 首选方式
- 所有从此AssetBundle中加载的Object都将被销毁
- Scene中对这些Object的引用将会丢失
AssetBundle.UnloadAllAssetBundles()
- 卸载当前已加载的所有AssetBundle
- 参数作用与AssetBundle.Unload()方法的参数相同
Resources.UnloadUnusedAssets()
- 卸载所有未被使用的Assetzh
注意:Asset的清理会在特定时期触发,但也可以手动触发。当从AssetBundle中加载的Object被从活动的Scene中移除时,Unity不会自动将其卸载。卸载AssetBundle但不卸载从其中加载的Object将会破坏此AssetBundle与Object之间的链接关系。如果之后再次加载了此AssetBundle并加载同一Object,内存中就会产生新的Object副本,而不是使用未被卸载的那个Object。最好将那些需要同时加载或者更新的Object打包到同一个AssetBundle中。
(2)AB包管理
选择AssetBundle压缩方式和资源分配要合理
- 加载时间:从本地加载未压缩的AssetBundle要比加载压缩过的AssetBundle快得多
- 构建时间:LZMA和LZ4压缩文件的速度非常慢
- 应用大小:如果AssetBundle被附带在项目中,将它们压缩可以减小应用程序的体积
- 内存占用:如果考虑内存占用量,请使用不压缩或者以LZ4压缩的AssetBundle
- 下载时间:如果AssetBundle很大或者用户网络带宽有限,那可能需要压缩
- 如果AssetBundle中的主要内容是使用紧密压缩算法进行压缩的DXT压缩纹理,那么这个AssetBundle不应该再被压缩
- 强烈建议开发者不要在WebGL项目中使用压缩的AssetBundle
(3)AB包加载代码
private IEnumerator RequestObjectFromServerAsynchronously(string ItemResPath, System.Action<GameObject> callBack)
{
int startindex = ItemResPath.IndexOf("webapps/") + 8;
string url = UserInfoModel.Instance().userUpLoadServerUrl + "/" + ItemResPath.Substring(startindex, ItemResPath.Length - startindex);
WWW assetwww = new WWW(url);
yield return assetwww;
if (!string.IsNullOrEmpty(assetwww.error))
{
yield break;
}
AssetBundle bundle = assetwww.assetBundle;
// AssetBundleCreateRequest request = AssetBundle.LoadFromMemoryAsync(assetwww.bytes);
AssetBundleRequest request = bundle.LoadAssetAsync(ItemResPath.Substring(ItemResPath.LastIndexOf('/') + 1).Replace(".asset", ""));
yield return request;
// AssetBundle bundle = request.assetBundle;
if (bundle == null)
{
yield break;
}
GameObject @object = request.asset as GameObject;
//加载出object,对其进行操作
bundle.Unload(false);
if (callBack != null)
callBack(@object);
}
注意:AB包内的资源需要通过AssetBundle.Load()来加载到内存中。
- 对于GameObject来说,通常情况下需要对其进行改动,所以它是完全复制一份该资源来进行的实例化。也就是说,当AB包中的GameObject从内存中卸载后,实例化的GameObject不会因此丢失。并且对实例化对象的修改不会影响到GameObject资源。
- 对于Shader和Texture来说,通常情况下不需要对其进行改动,所以它是通过引用来进行的实例化。也就是说,当AB包中的Shader和Texture资源从内存中卸载后,实例化的Shader和Texture会出现资源丢失的情况。并且对实例化对象的修改会影响到Shader和Texture资源。
- 对于Material和Mesh来说,有时候可能需要对其进行改动,所以它是通过引用+复制来进行的实例化。也就是说,当AB包中的Material和Mesh资源从内存中卸载后,实例化的Material和Mesh会出现资源丢失的情况。并且对实例化对象的修改不会影响到Material和Mesh资源。
4.AssetBundle卸载
由于大多数项目允许用户重复体验内容,因此了解何时加载或卸载资产绑定非常重要。如果未正确卸载AssetBundle,可能会导致内存中的对象重复。在某些情况下,不正确地卸载资产绑定也可能导致不良行为,例如导致纹理丢失。
如O是某资源,如果调用AssetBundle.Unload(true),那么M将从场景中移除、销毁并卸载。但是,如果调用AssetBundle.Unload(false),则AB的头信息将被卸载,但O将保留在场景中,并且仍然可以正常工作。调用AssetBundle.Unload(false)会中断O和AB之间的链接。如果以后再次加载AB,则AB中包含的对象的新副本将加载到内存中。如果以后再次加载AB,则会重新加载资产绑定头信息的新副本。但是,O不是从AB的新副本加载的。Unity不会在AB的新副本和O之间建立任何链接。 如果调用AssetBundle.LoadAsset()重新加载O,Unity将不会将O的旧副本解释为AB中数据的实例。因此,Unity将加载O的新副本,并且场景中将有两个相同的O副本。
注意:然而对于大多数项目而言,这种行为是不可取的。大多数项目应该使用AssetBundle.Unload(true)并采用一种方法来确保对象不重复。两种常用方法是:
- (1)在应用程序的生命周期中设置一些卸载资源绑定的节点,例如场景切换时。
- (2)维护单个对象的引用计数,并仅在其所有组成对象都未使用时卸载资产绑定。这允许应用程序卸载和重新加载单个对象,而无需复制内存。
如果应用程序必须使用AssetBundle.Unload(false),则只能通过两种方式卸载单个对象:
- (1)消除对不需要的对象的所有引用。然后调用Resources.UnloadUnusedAssets。
- (2)以非添加方式加载场景(这里如果不懂的话可以考虑看下场景加载模式)。这将销毁当前场景中的所有对象并自动调用Resources.UnloadUnusedAssets。
5.CRC校验
AB包加载资源的完整方法实际上是AssetBundle.LoadFromFile(string path, uint crc, ulong offset),三个参数。其中第二个参数就是CRC校验符。
每个AB包的.manifest文件中也有CRC校验符,用于校验数据完整性。
6.总结
AssetBundle目前来说用户使用自由度高,用起来方便,但是从2020版官方出现了Addressables资源打包的方式。
AssetBundle之间依赖关系复杂,要合理规划使用,AssetBundle之间的依赖关系使用两个不同的API自动跟踪,具体取决于运行时环境。在Unity Editor中,可以通过AssetDatabase API 查询AssetBundle依赖项。可以通过AssetImporter API 访问和更改AssetBundle分配和依赖项。在运行时,Unity提供了一个可选API,用于通过基于ScriptableObject的AssetBundleManifest API 加载在AssetBundle构建期间生成的依赖关系信息。
7.插件AssetBundleBrowser
针对AssetBundle退出了第三方插件:AssetBundleBrowser
安装:Unity 2019版本可以直接在PackageManager里面找到此插件并直接安装,其他版本可在https://github.com/Unity-Technologies/AssetBundles-Browser 获取。将下载后的安装包解压到Unity工程的Packages文件夹下 (一定要解压)
正确获取到并安装完插件后,通过 Windows ----> AssetBundle Browser下打开AB包管理面板 一共有三个面板
- Configure面板 :能查看当前AB包及其内部资源的基本情况(大小,资源,依赖情况等)
- Build面板:负责AssetBundle打包的相关设置 按Build即可进行打包
- Inspect面板:主要用来查看已经打包后的AB包文件的一些详细情况(大小,资源路径等)
AB管理脚本
using System;
using System.Net.Mime;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
namespace Common
{
/// <summary>
/// AB包管理器 全局唯一 使用单例模式
/// </summary>
public class ABManager
{
//AB包缓存---解决AB包无法重复加载的问题 也有利于提高效率。
private Dictionary<string, AssetBundle> abCache;
private AssetBundle mainAB = null; //主包
private AssetBundleManifest mainManifest = null; //主包中配置文件---用以获取依赖包
//各个平台下的基础路径 --- 利用宏判断当前平台下的streamingAssets路径
private string basePath { get
{
//使用StreamingAssets路径注意AB包打包时 勾选copy to streamingAssets
#if UNITY_EDITOR || UNITY_STANDALONE
return Application.dataPath + "/StreamingAssets/";
#elif UNITY_IPHONE
return Application.dataPath + "/Raw/";
#elif UNITY_ANDROID
return Application.dataPath + "!/assets/";
#endif
}
}
//各个平台下的主包名称 --- 用以加载主包获取依赖信息
private string mainABName
{
get
{
#if UNITY_EDITOR || UNITY_STANDALONE
return "StandaloneWindows";
#elif UNITY_IPHONE
return "IOS";
#elif UNITY_ANDROID
return "Android";
#endif
}
}
//继承了单例模式提供的初始化函数
protected override void Init()
{
base.Init();
//初始化字典
abCache = new Dictionary<string, AssetBundle>();
}
//加载AB包
private AssetBundle LoadABPackage(string abName)
{
AssetBundle ab;
//加载ab包,需一并加载其依赖包。
if (mainAB == null)
{
//根据各个平台下的基础路径和主包名加载主包
mainAB = AssetBundle.LoadFromFile(basePath + mainABName);
//获取主包下的AssetBundleManifest资源文件(存有依赖信息)
mainManifest = mainAB.LoadAsset<AssetBundleManifest>("AssetBundleManifest");
}
//根据manifest获取所有依赖包的名称 固定API
string[] dependencies = mainManifest.GetAllDependencies(abName);
//循环加载所有依赖包
for (int i = 0; i < dependencies.Length; i++)
{
//如果不在缓存则加入
if (!abCache.ContainsKey(dependencies[i]))
{
//根据依赖包名称进行加载
ab = AssetBundle.LoadFromFile(basePath + dependencies[i]);
//注意添加进缓存 防止重复加载AB包
abCache.Add(dependencies[i], ab);
}
}
//加载目标包 -- 同理注意缓存问题
if (abCache.ContainsKey(abName)) return abCache[abName];
else
{
ab = AssetBundle.LoadFromFile(basePath + abName);
abCache.Add(abName, ab);
return ab;
}
}
//==================三种资源同步加载方式==================
//提供多种调用方式 便于其它语言的调用(Lua对泛型支持不好)
#region 同步加载的三个重载
/// <summary>
/// 同步加载资源---泛型加载 简单直观 无需显示转换
/// </summary>
/// <param name="abName">ab包的名称</param>
/// <param name="resName">资源名称</param>
public T LoadResource<T>(string abName,string resName)where T:Object
{
//加载目标包
AssetBundle ab = LoadABPackage(abName);
//返回资源
return ab.LoadAsset<T>(resName);
}
//不指定类型 有重名情况下不建议使用 使用时需显示转换类型
public Object LoadResource(string abName,string resName)
{
//加载目标包
AssetBundle ab = LoadABPackage(abName);
//返回资源
return ab.LoadAsset(resName);
}
//利用参数传递类型,适合对泛型不支持的语言调用,使用时需强转类型
public Object LoadResource(string abName, string resName,System.Type type)
{
//加载目标包
AssetBundle ab = LoadABPackage(abName);
//返回资源
return ab.LoadAsset(resName,type);
}
#endregion
//================三种资源异步加载方式======================
/// <summary>
/// 提供异步加载----注意 这里加载AB包是同步加载,只是加载资源是异步
/// </summary>
/// <param name="abName">ab包名称</param>
/// <param name="resName">资源名称</param>
public void LoadResourceAsync(string abName,string resName, System.Action<Object> finishLoadObjectHandler)
{
AssetBundle ab = LoadABPackage(abName);
//开启协程 提供资源加载成功后的委托
StartCoroutine(LoadRes(ab,resName,finishLoadObjectHandler));
}
private IEnumerator LoadRes(AssetBundle ab,string resName, System.Action<Object> finishLoadObjectHandler)
{
if (ab == null) yield break;
//异步加载资源API
AssetBundleRequest abr = ab.LoadAssetAsync(resName);
yield return abr;
//委托调用处理逻辑
finishLoadObjectHandler(abr.asset);
}
//根据Type异步加载资源
public void LoadResourceAsync(string abName, string resName,System.Type type, System.Action<Object> finishLoadObjectHandler)
{
AssetBundle ab = LoadABPackage(abName);
StartCoroutine(LoadRes(ab, resName,type, finishLoadObjectHandler));
}
private IEnumerator LoadRes(AssetBundle ab, string resName,System.Type type, System.Action<Object> finishLoadObjectHandler)
{
if (ab == null) yield break;
AssetBundleRequest abr = ab.LoadAssetAsync(resName,type);
yield return abr;
//委托调用处理逻辑
finishLoadObjectHandler(abr.asset);
}
//泛型加载
public void LoadResourceAsync<T>(string abName, string resName, System.Action<Object> finishLoadObjectHandler)where T:Object
{
AssetBundle ab = LoadABPackage(abName);
StartCoroutine(LoadRes<T>(ab, resName, finishLoadObjectHandler));
}
private IEnumerator LoadRes<T>(AssetBundle ab, string resName, System.Action<Object> finishLoadObjectHandler)where T:Object
{
if (ab == null) yield break;
AssetBundleRequest abr = ab.LoadAssetAsync<T>(resName);
yield return abr;
//委托调用处理逻辑
finishLoadObjectHandler(abr.asset as T);
}
//====================AB包的两种卸载方式=================
//单个包卸载
public void UnLoad(string abName)
{
if(abCache.ContainsKey(abName))
{
abCache[abName].Unload(false);
//注意缓存需一并移除
abCache.Remove(abName);
}
}
//所有包卸载
public void UnLoadAll()
{
AssetBundle.UnloadAllAssetBundles(false);
//注意清空缓存
abCache.Clear();
mainAB = null;
mainManifest = null;
}
}
}