using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Events;
using Object = UnityEngine.Object;
/// <summary>
/// 知识点
/// 1.AB包相关的API
/// 2.单例模式
/// 3.委托 --->Lambda表达式
/// 4.协程
/// 5.字典
/// </summary>
public class ABMgr : SingletonAutoMono<ABMgr>
{
//AB包管理器 目的是
//让外部更方便的进行资源加载
//主包
private AssetBundle mainAB = null;
//依赖包获取用的配置文件
private AssetBundleManifest manifest = null;
//AB包不能够重复加载 重复加载会报错
//字典 用字典来存储 加载过的AB包
private Dictionary<string, AssetBundle> abDic = new Dictionary<string, AssetBundle>();
/// <summary>
/// 这个AB包存放路径 方便修改
/// </summary>
private string PathUrl
{
get
{
return Application.streamingAssetsPath + "/";
}
}
private string MainABName
{
get
{
#if UNITY_IOS
return "IOS";
#elif UNITY_ANDROID
return "Android";
#elif UNITY_WEBGL
return "WebGL";
#else
return "PC";
#endif
}
}
//同步加载 不指定类型
public Object LoadRes(string abName, string resName)
{
//加载AB包
LoadAB(abName);
//为了外面方便 在加载资源时 判断一下 资源是不是Gameobject
//如果是 直接实例化了 再返回给外部
Object obj = abDic[abName].LoadAsset(resName);
if (obj is GameObject)
return Instantiate(obj);
else
return obj;
}
//同步加载 根据type指定类型
public Object LoadRes(string abName, string resName, System.Type type)
{
//加载AB包
LoadAB(abName);
//为了外面方便 在加载资源时 判断一下 资源是不是Gameobject
//如果是 直接实例化了 再返回给外部
Object obj = abDic[abName].LoadAsset(resName,type);
if (obj is GameObject)
return Instantiate(obj);
else
return obj;
}
//同步加载 根据泛型指定类型
public T LoadRes<T>(string abName, string resName) where T:Object
{
//加载AB包
LoadAB(abName);
//为了外面方便 在加载资源时 判断一下 资源是不是Gameobject
//如果是 直接实例化了 再返回给外部
T obj = abDic[abName].LoadAsset<T>(resName);
if (obj is GameObject)
return Instantiate(obj);
else
return obj;
}
//异步加载的方法
//这里的异步加载 AB包并没有使用异步加载
//只是从AB包中 加载资源时 使用异步
//根据名字异步加载资源
public void LoadResAsync(string abName, string resName, UnityAction<Object> callBack)
{
//加载AB包
LoadAsyncAB(abName, () => {
StartCoroutine(ReallyLoadResAsync(abName, resName, callBack));
});
}
private IEnumerator ReallyLoadResAsync(string abName, string resName, UnityAction<Object> callBack)
{
//为了外面方便 在加载资源时 判断一下 资源是不是Gameobject
//如果是 直接实例化了 再返回给外部
AssetBundleRequest abr = abDic[abName].LoadAssetAsync(resName);
yield return abr;
//异步加载结束后 通过委托 传递给外部 外部来使用
if (abr.asset is GameObject)
callBack(Instantiate(abr.asset)) ;
else
callBack(abr.asset);
}
//根据Type异步加载资源
public void LoadResAsync(string abName, string resName, System.Type type, UnityAction<Object> callBack)
{
//加载AB包
LoadAsyncAB(abName, () => {
StartCoroutine(ReallyLoadResAsync(abName, resName, type, callBack));
});
}
private IEnumerator ReallyLoadResAsync(string abName, string resName, System.Type type, UnityAction<Object> callBack)
{
//为了外面方便 在加载资源时 判断一下 资源是不是Gameobject
//如果是 直接实例化了 再返回给外部
AssetBundleRequest abr = abDic[abName].LoadAssetAsync(resName,type);
yield return abr;
//异步加载结束后 通过委托 传递给外部 外部来使用
if (abr.asset is GameObject)
callBack(Instantiate(abr.asset));
else
callBack(abr.asset);
}
//根据泛型 异步加载资源
public void LoadResAsync<T>(string abName, string resName, UnityAction<T> callBack) where T : Object
{
//加载AB包
LoadAsyncAB(abName, () =>
{
StartCoroutine(ReallyLoadResAsync<T>(abName, resName, callBack));
});
}
private IEnumerator ReallyLoadResAsync<T>(string abName, string resName, UnityAction<T> callBack) where T : Object
{
//为了外面方便 在加载资源时 判断一下 资源是不是Gameobject
//如果是 直接实例化了 再返回给外部
AssetBundleRequest abr = abDic[abName].LoadAssetAsync<T>(resName);
yield return abr;
//异步加载结束后 通过委托 传递给外部 外部来使用
if (abr.asset is GameObject)
callBack(Instantiate(abr.asset) as T);
else
callBack(abr.asset as T);
}
//单个包卸载
public void Unload(string abName)
{
if (abDic.ContainsKey(abName))
{
abDic[abName].Unload(false);
abDic.Remove(abName);
}
}
//所有包的卸载
public void ClearAB()
{
AssetBundle.UnloadAllAssetBundles(false);
abDic.Clear();
mainAB = null;
manifest = null;
}
/// <summary>
/// 加载AB包
/// </summary>
/// <param name="abName"></param>
public void LoadAB(string abName)
{
//加载AB包
if (mainAB == null)
{
mainAB = AssetBundle.LoadFromFile(PathUrl + MainABName);
manifest = mainAB.LoadAsset<AssetBundleManifest>("AssetBundleManifest");
}
//获取依赖包相关信息
AssetBundle ab;
string[] strs = manifest.GetAllDependencies(abName);
for (int i = 0; i < strs.Length; i++)
{
//判断包是否加载过
if (!abDic.ContainsKey(strs[i]))
{
ab = AssetBundle.LoadFromFile(PathUrl + strs[i]);
abDic.Add(strs[i], ab);
}
}
//加载资源来源包
//如果没有加载过 再加载
if (!abDic.ContainsKey(abName))
{
ab = AssetBundle.LoadFromFile(PathUrl + abName);
abDic.Add(abName, ab);
}
}
/// <summary>
/// 异步加载AB包
/// </summary>
/// <param name="abName"></param>
public void LoadAsyncAB(string abName, UnityAction callback)
{
StartCoroutine(ReallyLoadAsyncAB(abName, callback));
}
private bool isRealed = false;
private IEnumerator ReallyLoadAsyncAB(string abName , UnityAction callback)
{
//加载AB包
AssetBundleCreateRequest abcr = null;
//判断如果为null并且某有人在加载它 才会去加载
if (mainAB == null && !isRealed)
{
isRealed = true;
abcr = AssetBundle.LoadFromFileAsync(PathUrl + MainABName);
yield return abcr;
mainAB = abcr.assetBundle;
AssetBundleRequest abr = mainAB.LoadAssetAsync<AssetBundleManifest>("AssetBundleManifest");
yield return abr;
manifest = (AssetBundleManifest)abr.asset;
}
//如果有人在加载 等别人加载完了 再说
else
{
//每帧都检测别人加载完没有
while(true)
{
//如果这两个不为空了 证明别人已经把他们加载好了 我们就可以跳出循环 继续做下面的逻辑了
if (mainAB != null && manifest != null)
break;
yield return null;
}
}
//获取依赖包相关信息
string[] strs = manifest.GetAllDependencies(abName);
for (int i = 0; i < strs.Length; i++)
{
//判断包是否加载过 如果从来没有加载过 那么我们去加载
if (!abDic.ContainsKey(strs[i]))
{
//先占个位置
abDic.Add(strs[i], null);
abcr = AssetBundle.LoadFromFileAsync(PathUrl + strs[i]);
yield return abcr;
//真正加载结束后 记录加载完成的ab包
abDic[strs[i]] = abcr.assetBundle;
}
//字典里面有记录 那么可能字典里面也是空的
else
{
//每帧都判断 这个包真正加载好没有 如果为null 表示没有加载好 我们就继续等 等到不为null循环自然就停了
while(abDic[strs[i]] == null)
{
yield return null;
}
}
}
//这下面的逻辑 同理
//加载资源来源包
//如果没有加载过 再加载
if (!abDic.ContainsKey(abName))
{
abDic.Add(abName, null);
abcr = AssetBundle.LoadFromFileAsync(PathUrl + abName);
yield return abcr;
abDic[abName] = abcr.assetBundle;
}
//字典里面有记录 那么可能字典里面也是空的
else
{
//每帧都判断 这个包真正加载好没有 如果为null 表示没有加载好 我们就继续等 等到不为null循环自然就停了
while (abDic[abName] == null)
{
yield return null;
}
}
callback();
}
}
使用:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class ABTest : MonoBehaviour
{
public Image img;
// Start is called before the first frame update
void Start()
{
//GameObject obj = ABMgr.GetInstance().LoadRes("model", "Cube", typeof(GameObject)) as GameObject;
//obj.transform.position = -Vector3.up;
//GameObject obj2 = ABMgr.GetInstance().LoadRes<GameObject>("model", "Cube");
//obj2.transform.position = Vector3.up;
ABMgr.GetInstance().LoadResAsync<GameObject>("model", "Cube",(obj)=>
{
obj.transform.position = -Vector3.up;
});
ABMgr.GetInstance().LoadResAsync<GameObject>("model", "Cube", (obj) =>
{
obj.transform.position = Vector3.up;
});
//关于AB包的依赖 --- 一个资源身上用到了别的AB包中的资源
//通过它创建对象 会出现资源丢失的情况
//这种时候 需要把依赖包 一起加载了 才能正常
//第一步 加载AB包
//AssetBundle ab = AssetBundle.LoadFromFile(Application.streamingAssetsPath + "/model");
////依赖包的关键知识点 --- 利用主包 获取依赖信息
////1.加载主包
//AssetBundle abMain = AssetBundle.LoadFromFile(Application.streamingAssetsPath + "/PC");
////2.加载主包中的固定文件
//AssetBundleManifest abManiFest = abMain.LoadAsset<AssetBundleManifest>("AssetBundleManifest");
////3.从固定文件中 得到依赖信息
//string[] strs = abManiFest.GetAllDependencies("model");
////得到了 依赖包的名字
//for (int i = 0; i < strs.Length; i++)
//{
// AssetBundle.LoadFromFile(Application.streamingAssetsPath + "/" + strs[i]);
//}
////第二步 加载 AB包中的资源
////只使用名字加载 会出现 同名不同类型资源 分不清
////建议大家用 泛型加载 或者 时 Type指定类型
//GameObject obj = ab.LoadAsset<GameObject>("Cube");
////GameObject obj = ab.LoadAsset("Cube", typeof(GameObject)) as GameObject;
//Instantiate(obj);
//ab.Unload(false);
//AB包不能重复加载 否则报错
//AssetBundle ab2 = AssetBundle.LoadFromFile(Application.streamingAssetsPath + "/model");
//加载一个圆
//GameObject obj2 = ab.LoadAsset("Sphere", typeof(GameObject)) as GameObject;
//Instantiate(obj2);
//异步加载--->协程
//StartCoroutine(LoadABRes("head", "body14601_stand_3_0"));
}
IEnumerator LoadABRes( string ABName, string resName)
{
//第一步 加载AB包
AssetBundleCreateRequest abcr = AssetBundle.LoadFromFileAsync(Application.streamingAssetsPath + "/" + ABName);
yield return abcr;
//第二步 加载资源
AssetBundleRequest abq = abcr.assetBundle.LoadAssetAsync(resName, typeof(Sprite));
yield return abq;
//abq.asset as Sprite;
img.sprite = abq.asset as Sprite;
}
// Update is called once per frame
void Update()
{
if (Input.GetKeyDown(KeyCode.Space))
{
//卸载所有加载的AB包 参数为true 会把通过AB包加载的资源也卸载
AssetBundle.UnloadAllAssetBundles(false);
}
}
}