【Unity3D】URP中的打包资源加载

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了【Unity3D】URP中的打包资源加载相关的知识,希望对你有一定的参考价值。

参考技术A   当项目打包后,Shader.Find寻找Shader的方式可能会失效,原因是当前项目可能没有引用此Shader。
  一般解决方式,是Edit/Project Settings/Graphics,将Shader加入到Always Includes Shaders中,我不喜欢这种方法,因为Always Includes Shaders一般是加载那些Hidden Shader的。所以我是新建一个材质球引用Shader,并把材质球拖到需要用到的位置。
  值得注意的是,URP中的Shader也能在打包后构建材质,我找到Shader的引用位置:

  用Reload标注的地址,这个路径基于URP Package地址。
  点进去能看到ReloadAttribute,就是把路径存储起来。
  ShaderResources类被PostProcessData引用:

  这是一个ScriptableObject,对象可以被存储为asset文件。
  存储、加载过程是在ForawdRendererData中,ForwardRendererData本身也是一个ScriptableObject,它包含了一个PostProcessData对象,当Create时,它会先从文件中读取自己的属性,然后再从文件读取PostProcessData对象的属性:

  ResourceReloader.TryReloadAllNullIn间接调用ReloadAllNullIn:

  当ForwardRendererData调用 ResourceReloader.TryReloadAllNullIn(this…… 时,先对ForwardRendererData对象中的每个有Reload的成员赋值、寻找路径,其中postProcessData成员是这样的:

  PostProcessData.asset中存储了后处理需要的Shader和纹理,直接点看不见,是因为PostProcessDataEditor.cs中进行了开发模式判断,注释掉就好:

  能看到:

AssetBundle资源打包与加载

AssetBundle资源打包

技术图片

 1.AssetLabels资源标签

  文件名:资源打包成AssetBundle后的文件名,类似于压缩包的名字
  后缀:自定义
  文件名和后缀名都是小写格式(大写会自动转为小写)
2. BuildPipeline.BuildAssetBundles(string outputPath, BuildAssetBundleOptions assetBundleOptions, BuildTarget targetPlatform) 打包所有设置了AssetLabel的资源
  outputPath:路径,打包出来的AssetBunlde文件存放的位置
  BuildAssetBundleOptions:选项,设置AssetBundle打包过程中压缩方式

    BuildAssetBundleOptions枚举选项:

      None使用LZMA压缩算法进行压缩,打包后资源体积最小
      UncompressedAssetBundle不压缩,打包后的AssetBundle体积最大,但是加载速度最快
      ChunkBasedCompression使用LZ4压缩算法进行压缩,打包后的AssetBundle体积和加载速度介于上面二者之间
  BuildTarget:平台,AssetBundle是平台之间不兼容的,IOS,Android是两套资源

  
在AssetLabels区域填写AssetBundle名称的时候,名称是可以分目录嵌套的:文件夹名/文件名

 

AssetBundle打包后的资源包,分两部分组成:

  1.资源打包出来的AssetBundle文件
  2.AssetBundle文件配套的manifest文本文件

技术图片

manifest文件

manifest文件用于专门存储打包后的AssetBundle文件的基本信息,主要包含:
  CRC校验码:类似于MD5,用于计算出该资源的一个特殊信息标示
  ClassTypes列表:当前资源关联使用到了Unity中的哪些类,这些类是以编号索引的形式存在的,每个编号都对应一个类文件
  Assets:AssetsBundle里包含了哪些资源文件
  Dependencies:依赖

技术图片

 在打包出来的AssetBundle文件中,有一个特殊的manifest文件,和AssetBundle存放的文件夹同名,且只在根文件夹下有唯一的一个

这个manifest文件可以称作"AssetBundle目录文件",它存储了打包出来的所有AssetBundle的文件的索引信息
通过这个目录文件,可以找到所有的AssetBundle文件

 

AssetBundle资源加载

将项目资源打包成AssetBundle后,一般有两种操作

一.将这些AssetBundle留着项目工程中,当成普通资源使用

  1.加载AssetBundle资源到内存

    AssetBundle ab =  AssetBundle.LoadFromFile("路径");加载AssetBundle资源到内存,返回一个AssetBundle对象

        2.从AssetBundle中获取资源

    T res = ab.LoadAsset<T>("资源名");

   3.实例化

打包

[MenuItem("xx/xxx")] 
public static void Build_WINDOWS64()
{
  string outputPath = Path.Combine(Application.streamingAssetsPath, "Data");
  if (!Directory.Exists(outputPath))
    Directory.CreateDirectory(outputPath);

  BuildPipeline.BuildAssetBundles(outputPath, BuildAssetBundleOptions.None, BuildTarget.StandaloneWindows64);//自己选择平台
  AssetDatabase.Refresh();
}

加载

using System.Collections;
using System.Collections.Generic;

public class AssetBundleManager
{
    private readonly string _path = UnityEngine.Application.streamingAssetsPath + "/Data/";
    private static AssetBundleManager _instance;
    private bool mIsLoadingEnd = false;
    public bool IsLoadingEnd { get { return mIsLoadingEnd; } }
    private List<string> mLoadingNames;
    private Dictionary<string, UnityEngine.Object> mAssets;
    private Dictionary<string, UnityEngine.Texture2D> mTexture2Ds;

    public static AssetBundleManager GetInstance()
    {
        if (_instance == null)
        {
            _instance = new AssetBundleManager();
        }
        return _instance;
    }
    
    private AssetBundleManager()
    {
        mLoadingNames = new List<string>(){
            //todo:要加载的资源名
        };

        mAssets = new Dictionary<string, UnityEngine.Object>();
        mTexture2Ds = new Dictionary<string, UnityEngine.Texture2D>();
    }
    
    /// <summary>
    /// 加载assetsbundle
   /// 记得先加载到内存StartCoroutine(AssetBundleManager.GetInstance().Loading());
/// </summary> public IEnumerator Loading() { foreach (var nameAndType in mLoadingNames) { string filename = _path + nameAndType.ToLower(); UnityEngine.AssetBundleCreateRequest request = UnityEngine.AssetBundle.LoadFromFileAsync(filename); yield return request; LoadAsset(request.assetBundle); } mIsLoadingEnd = true; } private UnityEngine.AssetBundle LoadAssetBundle(string fileName) { UnityEngine.AssetBundle AB = UnityEngine.AssetBundle.LoadFromFile(_path + fileName); return AB; } private void LoadAsset(UnityEngine.AssetBundle ab) { UnityEngine.Object[] assets = ab.LoadAllAssets(); for (int i = 0; i < assets.Length; i++) { //一张图片的AssetsBundle包含自身Texture2D和它里面全部Sprite,有可能重名 if (!typeof(UnityEngine.Texture2D).Equals(assets[i].GetType())) mAssets.Add(assets[i].name, assets[i]); else mTexture2Ds.Add(assets[i].name, assets[i] as UnityEngine.Texture2D); } } /// <summary> /// 从assetbundle获取texture2d /// </summary> public UnityEngine.Texture2D GetTexture2D(string name) { UnityEngine.Texture2D obj; mTexture2Ds.TryGetValue(name, out obj); return obj; } /// <summary> /// 从assetbundle获取Object /// </summary> public UnityEngine.Object GetAsset(string name) { UnityEngine.Object obj; mAssets.TryGetValue(name, out obj); return obj; } }

二.将这些AssetBundle上传服务器,客户端第一次运行时从服务器下载AssetBundle缓存到本地再使用(实现客户端的安装包与资源分离,降低客户端安装包的体积)

 1.服务器端下载主文件

    从服务器端下载AssetBundle需要web路径地址,每一个地址对应一个AssetBundle文件.我们不可能在代码中写几十上百个文件的web地址,所以我们需要先下载"目录AssetBundle文件",然后通过它来间接获取其他AssetBundle文件的下载路径地址.

需要引入命名空间using UnityEngine.Networking;

using System.Collections;
using System.IO;
using UnityEngine;
using UnityEngine.Networking;

public class TestAssetBundle : MonoBehaviour
{
    private string mainAssetBundleURL = @"http://www.xxx.com.xxx/AssetBundleFile";
    private string allAssetBundleURL =  @"http://www.xxx.com.xxx/";

    void Start()
    {
        StartCoroutine(DownloadMainAssetBundle());
    }

    /// <summary>
    /// 下载目录AssetBundle文件
    /// </summary>
    IEnumerator DownloadMainAssetBundle()
    {
        //创建一个获取AssetBundle文件的web请求
        UnityWebRequest request = UnityWebRequestAssetBundle.GetAssetBundle(mainAssetBundleURL);

        //发送web请求
        yield return request.SendWebRequest();

        //从请求中获取内容,返回AssetBundle类型的数据
        AssetBundle ab = DownloadHandlerAssetBundle.GetContent(request);

        //从"目录AssetBundle"中获取manifest数据
        AssetBundleManifest manifest = ab.LoadAsset<AssetBundleManifest>("AssetBundleManifest");

        //获取manifest文件中所有的AssetBundle的名称信息
        string[] names = manifest.GetAllAssetBundles();

        foreach (var name in names)
        {
            StartCoroutine(DownloadAssetBundleAndSave(allAssetBundleURL + name));
        }
    }
    
    /// <summary>
    /// 下载AssetBundle并保存到本地
    /// </summary>
    IEnumerator DownloadAssetBundleAndSave(string url)
    {
        UnityWebRequest request = UnityWebRequest.Get(url);
        yield return request.SendWebRequest();
        // 截取路径地址中的文件名(需要引入System.IO)
        string fileName = Path.GetFileName(url);
        SaveAssetBundle(fileName, request.downloadHandler.data);
    }

    /// <summary>
    /// 存储AssetBundle为本地文件
    /// </summary>
    private void SaveAssetBundle(string fileName, byte[] bytes)
    {
        //创建一个文件信息对象(System.IO)
        FileInfo fileInfo = new FileInfo(Application.streamingAssetsPath + "//" + fileName);

        //创建一个文件流对象
        FileStream fs = fileInfo.Create();

        //通过文件流对象,往文件内写入信息
        fs.Write(bytes, 0, bytes.Length);

        //文件写入存储到硬盘
        fs.Flush();

        //关闭文件流对象
        fs.Close();

        //销毁文件对象
        fs.Dispose();
    }
}

 

以上是关于【Unity3D】URP中的打包资源加载的主要内容,如果未能解决你的问题,请参考以下文章

Unity3D之Mecanim动画系统学习笔记:Mecanim动画的资源加载相关

unity3D中动态加载物体的常用的方法

Assetbundle 打包加载及服务器加载等(采用unity3d5.0后的新版)

Unity3D资源异步加载——Addressables资源异步加载

Unity3D资源异步加载——图片音视频资源加载

unity3D加密