1.AB包生成脚本
using UnityEngine;
using UnityEditor;
using System.IO;
using System;
using System.Collections.Generic;/// <summary>
/// AB包创建
/// </summary>
public class CreateAssetBundles : MonoBehaviour
{public static string BuildAssetBundlePath = Application.dataPath + "/AssetsPach/AssetBundles";public static void BuildAssetBundle_WeixinMiniGame(){string dir = BuildAssetBundlePath; //相对路径if (!Directory.Exists(dir)) //判断路径是否存在{Directory.CreateDirectory(dir);}BuildPipeline.BuildAssetBundles(dir, BuildAssetBundleOptions.None, BuildTarget.WeixinMiniGame); //这里是第一点注意事项,BuildTarget类型选择WebGLAssetDatabase.Refresh();Debug.Log("打包完成");}//需要打包的资源目录public static string SetAssetBundlePath = Application.dataPath + "/AssetsPach/WortAsset";public static void SetAssetBundle_WeixinMiniGame(){string dir = SetAssetBundlePath; //相对路径AssetDatabase.RemoveUnusedAssetBundleNames();//移除无用的AssetBundleName//Debug.LogError(Application.dataPath);//上级路径 F:/TUANJIEProject/My project/Assetslist_Files = new List<stru_FileInfo>();ContinueCheck(dir);for (int a = 0; a < list_Files.Count; a++)//{SetBundleName(list_Files[a].assetPath);}Debug.Log("生成ab包完成");//SetBundleName("Assets/Ship/AC_Enterprise_T01/prefab/AC_Enterprise_T01_M01_ShipMesh.prefab");}//******资源参数static List<stru_FileInfo> list_Files;//文件列表static string assetBundleName = "ab";static string assetBundleVariant = "";//int indentation;//缩进等级struct stru_FileInfo{public string fileName;public string filePath;//绝对路径public string assetPath;//U3D内部路径public Type assetType;}static void ContinueCheck(string path){DirectoryInfo directory = new DirectoryInfo(path);FileSystemInfo[] fileSystemInfos = directory.GetFileSystemInfos();//获取文件夹下的文件信息foreach (var item in fileSystemInfos){int idx = item.ToString().LastIndexOf(@"\");string name = item.ToString().Substring(idx + 1);if (!name.Contains(".meta"))//剔除meta文件{CheckFileOrDirectory(item, path + "/" + name);}}}static void CheckFileOrDirectory(FileSystemInfo fileSystemInfo,string path){FileInfo fileInfo = fileSystemInfo as FileInfo;if (fileInfo != null){stru_FileInfo t_file = new stru_FileInfo();t_file.fileName = fileInfo.Name;t_file.filePath = fileInfo.FullName;t_file.assetPath = "Assets" + fileInfo.FullName.Replace(Application.dataPath.Replace("/", "\\"), "");//用于下一步获得文件类型t_file.assetType = AssetDatabase.GetMainAssetTypeAtPath(t_file.assetPath);list_Files.Add(t_file);}else{ContinueCheck(path);}}static void SetBundleName(string path){var importer = AssetImporter.GetAtPath(path);string[] strs = path.Split(".");string[] dictors = strs[0].Split('/');if (importer){if (assetBundleVariant != ""){importer.assetBundleVariant = assetBundleVariant;}if (assetBundleName != ""){importer.assetBundleName = path.ToLower() + "." + assetBundleName;}}else{Debug.Log("importer是空的"+ path);//jpg png tga}}}
2.AB包管理类
搭建资源服务器使用(HFS软件(https://www.pianshen.com/article/54621708008/))
using System;
using System.Net.Mime;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Object = UnityEngine.Object;
using Cysharp.Threading.Tasks; // 首先需要引入UniTask插件
using UnityEngine.Networking;
using UnityEngine.U2D;//已经加载的AB类
public class LoadedAssetBundle
{public AssetBundle bundle;public string abName;public int referenceCount = 0;public void AddRefer(){referenceCount++;}public int Decrease(){referenceCount--;if (referenceCount <= 0) referenceCount = 0;return referenceCount;}}/// <summary>
/// AB包管理器 全局唯一 使用单例模式
///
/// 需要有的功能
/// 1.资源名索引ab包名 的字典
/// 2.私有方法 通过资源名下载AB包,同时下载一个AB包时 会同时等待,已经存AB缓存 以防多次下载同一个AB包
/// 3.通过资源名 加载 预制体加载范例
/// 4.通过图集名,图片名 加载 通过TexturePacker创建的图集,如果需要访问图集的精灵对象,可以通过子对象加载接口。
/// 5.需要常驻的AB包 比如字体库 通用图集 不需要卸载
///
///
/// </summary>
public class ABManager
{private static readonly Lazy<ABManager> lazy = new Lazy<ABManager>(() => new ABManager());public static ABManager Instance { get { return lazy.Value; } }private ABManager() { Init(); }//AB包缓存---解决AB包无法重复加载的问题 也有利于提高效率。private Dictionary<string, LoadedAssetBundle> abCache;private AssetBundle mainAB = null; //主包private AssetBundleManifest mainManifest = null; //主包中配置文件---用以获取依赖包private string basePath = "https://xxx/gh5_test/StreamingAssets/AssetBundles/";private string mainABName = "AssetBundles";//继承了单例模式提供的初始化函数void Init(){//初始化字典abCache = new Dictionary<string, LoadedAssetBundle>();}public Dictionary<string, string> AssetNameToABName = new Dictionary<string, string>();public async UniTask<GameObject> LoadAsset(string assetName){string abName = assetName.ToLower() + ".ab";AssetBundle ab = await LoadABPackage(abName);return ab.LoadAsset<GameObject>(assetName);}/// <summary>/// 加载图集里面的图片/// 案例/// Image a = nul;;/// if (a != null)/// ABManager.Instance.UnloadAsset(a);/// a = ABManager.Instance.LoadAtlasSprite(a);/// </summary>/// <param name="assetName"></param>/// <param name="textureName"></param>/// <returns></returns>public async UniTask<Sprite> LoadAtlasSprite(string assetName,string textureName){string abName = assetName.ToLower() + ".ab";AssetBundle ab = await LoadABPackage(abName);SpriteAtlas spriteAtlas = ab.LoadAsset<SpriteAtlas>(assetName); return spriteAtlas.GetSprite(textureName);}//单个包卸载public void UnloadAsset(string assetName){string abName = assetName.ToLower() + ".ab";//--引用if (abCache.ContainsKey(abName)){int reference = abCache[abName].Decrease();if (reference == 0){abCache[abName].bundle.Unload(true);//注意缓存需一并移除abCache.Remove(abName);}}}//加载AB包private async UniTask<AssetBundle> LoadABPackage(string abName){//加载ab包,需一并加载其依赖包。if (mainAB == null){//获取ab包内容mainAB = await DownloadABPackage(mainABName); //获取主包下的AssetBundleManifest资源文件(存有依赖信息)mainManifest = mainAB.LoadAsset<AssetBundleManifest>("AssetBundleManifest");}//加载目标包 -- 同理注意缓存问题if (abCache.ContainsKey(abName)){ abCache[abName].AddRefer(); //++引用return (abCache[abName].bundle);}else{//根据manifest获取所有依赖包的名称 固定APIstring[] dependencies = mainManifest.GetAllDependencies(abName);if (dependencies.Length > 0){var tasks = new List<UniTask>(); // 创建一个任务列表来存储异步操作//循环加载所有依赖包for (int i = 0; i < dependencies.Length; i++){//如果不在缓存则加入if (!abCache.ContainsKey(dependencies[i])) tasks.Add(LoadABPackage(dependencies[i]));}// 使用UniTask.WhenAll等待所有任务完成await UniTask.WhenAll(tasks);}await DownloadABPackage(abName);abCache[abName].AddRefer(); //++引用return (abCache[abName].bundle);}}//存儲下載操作Dictionary<string, UnityWebRequestAsyncOperation> ABRequest = new Dictionary<string, UnityWebRequestAsyncOperation>();async UniTask<AssetBundle> DownloadABPackage(string abname){if (abCache.ContainsKey(abname))return abCache[abname].bundle;string url = basePath + abname;Debug.Log(url);if (!ABRequest.ContainsKey(url)){UnityWebRequest req = UnityWebRequestAssetBundle.GetAssetBundle(url);UnityWebRequestAsyncOperation operation = req.SendWebRequest();ABRequest.Add(url, operation);}await ABRequest[url];if (!abCache.ContainsKey(abname)){AssetBundle ab = DownloadHandlerAssetBundle.GetContent(ABRequest[url].webRequest);abCache.Add(abname, new LoadedAssetBundle() { abName = abname, bundle = ab });ABRequest.Remove(abname);ABRequest[url].webRequest.Dispose();}return abCache[abname].bundle;}//所有包卸载public void UnLoadAll(){AssetBundle.UnloadAllAssetBundles(false);//注意清空缓存abCache.Clear();mainAB = null;mainManifest = null;}}
/**UnityWebRequest[AssetBundle].GetAssetBundle()
适用于从远程下载AssetBundle
能够缓存已下载的AssetBundle以便重用
缓存系统中的AssetBundle仅以文件名进行标识而不是使用URL
1.强烈建议开发者不要在WebGL项目中使用压缩的AssetBundle
2.注意:然而对于大多数项目而言,这种行为是不可取的。大多数项目应该使用AssetBundle.Unload(true)并采用一种方法来确保对象不重复。两种常用方法是:
(1)在应用程序的生命周期中设置一些卸载资源绑定的节点,例如场景切换时。
(2)维护单个对象的引用计数,并仅在其所有组成对象都未使用时卸载资产绑定。这允许应用程序卸载和重新加载单个对象,而无需复制内存。
3.ab包的卸載
如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副本。
4.CRC校验
AB包加载资源的完整方法实际上是AssetBundle.LoadFromFile(string path, uint crc, ulong offset),
三个参数。其中第二个参数就是CRC校验符。
每个AB包的.manifest文件中也有CRC校验符,用于校验数据完整性。
5.状态维护 AB加载的过程中 其他地方可能同时也在加载 需要维护AB的状态 加载中 加载完成 是否被使用
6.回调维护 同时多个加载AB包 不同地方的回调
7.在游戏中会存在大量的ab包资源,如果这些都加载在内存中,不卸载那么很有可能会造成游戏崩溃。就像一个容器只往里添加而不往外拿,
那么再大的容器也会被撑满。所以ab包什么时候卸载,卸载哪些包,就会是一个大问题。本文使用引用计数的方法解决ab包资源的管理问题。
(https://blog.csdn.net/qq_38399916/article/details/135643315)*/
3.使用案例
using Cysharp.Threading.Tasks;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Networking;public class GameEntrance : MonoBehaviour
{// Start is called before the first frame updateasync void Start(){GameObject go = await ABManager.Instance.LoadAsset("Assets/AssetsPach/Wortasset/Prefabs/Canvas.prefab");GameObject.Instantiate(go);}}