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)、所有其它包的相关信息(名称、依赖关系)Unity3D之资源管理——AB包管理详解_Unity3D

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;
    }

Unity3D之资源管理——AB包管理详解_AssetBundle_02

(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包及其内部资源的基本情况(大小,资源,依赖情况等)

Unity3D之资源管理——AB包管理详解_AssetBundle_03

  • Build面板:负责AssetBundle打包的相关设置 按Build即可进行打包

Unity3D之资源管理——AB包管理详解_Unity3D_04

  • Inspect面板:主要用来查看已经打包后的AB包文件的一些详细情况(大小,资源路径等)

Unity3D之资源管理——AB包管理详解_AssetBundle_05

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;
        }
    }
}