Unity-WebGL加载AB包

2024-07-12 1672阅读

一、前景回顾

上文写到PC、IOS、Android项目加载ab包资源,地址:

http://t.csdnimg.cn/RdB3i

二、WebGL打包AB包

打包的步骤和上文中的操作相同,唯一的不同就是在Build页面里Build Target选择WebGL。

Unity-WebGL加载AB包

三、WebGL加载AB包 

(1)内置渲染管线

当项目使用内置渲染管线时,所有材质的Shader为Standard。

Unity-WebGL加载AB包

(2)通用渲染管线URP

1、当项目为URP时,首先需要在Package Manager里导入Universal RP,如图

Unity-WebGL加载AB包

2、Create—Rendering—URP Assets(with Universal Render)来创建URP配置文件,如图

Unity-WebGL加载AB包

3、在Edit—Project Settings—Graphics,修改为刚才创建的文件。这一步操作完成后,场景中的物体材质可能会变成了洋红色。此时项目已由内置着色器转换为URP着色器。

Unity-WebGL加载AB包

4、Edit—Render Pipeline—Universal Render Pipeline—Upgrade Project Materials to UniversalRP Materials更新材质。 也可以手动修改材质的Shader为Universal Render Pipeline/Lit。如图

Unity-WebGL加载AB包

5、在软件中运行项目,加载出来的材质是洋红色的(不用担心)。

Unity-WebGL加载AB包

6、导出项目在本地浏览器加载,加载出来的模型材质是正常的。

Unity-WebGL加载AB包

四、加载ab包脚本

WebGL使用UnityWebRequest加载依赖和ab包
   
    //主包
    private AssetBundle abMain = null;
    //依赖
    private AssetBundleManifest abMainfest = null;
    //缓存字典,防止多次加载
    private Dictionary abDic = new Dictionary();
    /// 
    /// 平台对应的路径
    /// 
    private string PathURL = Application.streamingAssetsPath + "/";
    /// 
    /// 平台对应的主包名称
    /// 
    private string MainABName = "WebGL";
    //UnityWebRequest加载依赖
    private IEnumerator LoadDependences(string abName)
    {
        //加载主包
        if (abMain == null)
        {
            UnityWebRequest abRequest = UnityWebRequestAssetBundle.GetAssetBundle(PathURL + MainABName);
            yield return abRequest.SendWebRequest();
            abMain = DownloadHandlerAssetBundle.GetContent(abRequest);
            //获取主包下的AssetBundleManifest资源文件(存有依赖信息)
            abMainfest = abMain.LoadAsset("AssetBundleManifest");
        }
        //加载依赖
        AssetBundle ab = null;
        string[] dependences = abMainfest.GetAllDependencies(abName);
        if (dependences.Length > 0)
        {
            for (int i = 0; i  
测试
    
    public Transform[] bigRackPos;    
    public Transform[] smallRackPos;
    public void TestOne()
    {
        LoadByRequest("bigrack", "BigRack", (obj) =>
        {
            for (int i = 0; i 
        {
            for (int i = 0; i  

五、报错

在WebGL项目中加载ab包,不能同时加载加载多个ab包。
测试代码中,我写了两个方法来加载两个ab包,如果我把这两个方法写在Start方法里同时运行,就会报错并且第二个方法加载不出来。
    private void Start()
    {
        TestOne();
        TestTwo();
    }
错误显示
The AssetBundle 'G:/Unity Project/cgf/MyScene/NormalAssetBundle/Assets/StreamingAssets/WebGL' can't be loaded because another AssetBundle with the same files is already loaded.

六、解答

根据PC端加载ab包的思路来实现web端的加载,当同时加载多个ab包时,就会同时调用多次这个协程,就会出现冲突的情况,也就是上述这个问题。这里是显示的加载主包WebGL冲突的问题,大家可能也会遇到加载依赖包冲突的问题,针对目前出现的问题作出修改。

1、当我们对资源进行打包时,不再把相同的资源包打成其他包的依赖包,而是让每个包都独立,这样加载起来就舍弃了加载依赖的过程,也就避免了加载依赖包冲突的问题。打包如图

Unity-WebGL加载AB包Unity-WebGL加载AB包

 2、没了加载依赖包的操作也就不用加载主包了(加载主包是为了获取各个包之间的依赖关系,现在每个包都是独立的,不存在依赖了),这就删除掉之前的加载依赖和主包的协程方法LoadDependences即可。另外在加载协程了去掉 这句就可以了。

yield return StartCoroutine(LoadDependences(abName));

Unity-WebGL加载AB包

 3、完整代码
    //缓存字典,防止多次加载
    private Dictionary abDic = new Dictionary();
    /// 
    /// 平台对应的路径
    /// 
    private string PathURL = Application.streamingAssetsPath + "/";
   
    //UnityWebRequest加载
    public void LoadByRequest(string abName, string resName, UnityAction callBack) where T : Object
    {
        StartCoroutine(ReallyLoadByRequest(abName, resName, callBack));
    }
    private IEnumerator ReallyLoadByRequest(string abName, string resName, UnityAction callBack) where T : Object
    {
        if (!abDic.ContainsKey(abName))
        {
            UnityWebRequest abRequest3 = UnityWebRequestAssetBundle.GetAssetBundle(PathURL + abName);
            yield return abRequest3.SendWebRequest();
            AssetBundle ab = DownloadHandlerAssetBundle.GetContent(abRequest3);
            abDic[abName] = ab;
        }
        T loadedResource = abDic[abName].LoadAsset(resName);
        callBack(loadedResource as T);
    }
    //单个包卸载
    public void UnLoad(string abName)
    {
        if (abDic.ContainsKey(abName))
        {
            abDic[abName].Unload(false);
            //注意缓存需一并移除
            abDic.Remove(abName);
        }
    }
    //所有包卸载
    public void UnLoadAll()
    {
        AssetBundle.UnloadAllAssetBundles(true);
        //注意清空缓存
        abDic.Clear();
    }
4、测试
   private void Start()
    {
        TestOne();
        TestTwo();
    }
    public void TestOne()
    {
        LoadByRequest("bigrack", "BigRack", (obj) =>
        {
            for (int i = 0; i 
        {
            for (int i = 0; i  

同时加载多个ab包也没有问题了,已修复。欢迎大家测试留言!!!

VPS购买请点击我

免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们,邮箱:ciyunidc@ciyunshuju.com。本站只作为美观性配图使用,无任何非法侵犯第三方意图,一切解释权归图片著作权方,本站不承担任何责任。如有恶意碰瓷者,必当奉陪到底严惩不贷!

目录[+]