Unity AssetBundle 设置名称优化

原因

在大部分的 Unity 分析打包 AssetBundle 过程中,是采用对资源设置 AssetBundle 名称标签的方式来进行打包,类似如下:

public static void SetBundleName(string assetPath, string bundleName)
{
    var importer = AssetImporter.GetAtPath(assetPath);
    if (importer && importer.assetBundleName != bundleName)
    {
        importer.assetBundleName = bundleName;
    }
}

private static void BuildAssetBundlesInter(string outputPath, AssetBundleBuildOptions options)
{
    AssetBundleManifest manifest = BuildPipeline.BuildAssetBundles(outputPath,
                BuildAssetBundleOptions.UncompressedAssetBundle, GetBuildTarget());
}

每设置一个资源的 AssetBundle 名称,都会导致 Unity 写入其 .meta 文件,这样在分析的过程耗费不少时间。而且在打包完毕,又得重新设置所有资源的 AssetBundle 名称为空,否则仓库会提示有文件变化。

思路

修改流程,改成传入 AssetBundleBuild[] 的方式进行打包,这样就无需设置资源的 AssetBundle 名称。但是不能简单的把结果保存为 AssetBundleBuild[] 数组对象,因为打包工具支持当次不再分析资源,直接使用上一次的分析结果。所以要把 AssetBundleBuild[] 结果存为临时文件,方便下次使用,正常来说,这个临时文件一般只对当天的资源有效即可,当更新了资源,肯定得重新分析资源,不能再使用上次的结果。所以,只需要把这临时文件放在 Temp 目录下即可。

解决

创建一个独立类,来管理分析的 AssetBundle 构建信息:

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.Serialization.Formatters.Binary;
using UnityEditor;

/// <summary>
/// AssetBundle 分析映射
/// </summary>
public static class AssetBundleBuildMap
{
    private static AssetBundleMap assetBundleMap;

    public static void Clear()
    {
        if (assetBundleMap != null)
        {
            assetBundleMap = null;
        }
    }

    /// <summary>
    /// 普通资源设置 AssetBundle 名称接口
    /// </summary>
    /// <param name="assetPath"></param>
    /// <param name="assetBundleName"></param>
    public static void SetAssetPathAssetBundleName(string assetPath, string assetBundleName)
    {
        if (assetBundleMap == null)
        {
            assetBundleMap = new AssetBundleMap();
        }
        assetBundleMap.AddMap(assetPath, assetBundleName);
    }

    public static bool IsAssetPathHasAssetBundleName(string assetPath)
    {
        if (assetBundleMap == null)
        {
            return false;
        }
        return assetBundleMap.assetPathSet.Contains(assetPath);
    }

    public static List<string> GetAllAssetBundleNames()
    {
        if (assetBundleMap == null)
        {
            return null;
        }
        List<string> allAssetBundleNames = assetBundleMap.assetBundleDict.Keys.ToList();
        allAssetBundleNames.Sort(EditorUtility.NaturalCompare);
        return allAssetBundleNames;
    }

    public static List<string> GetAssetPathsFromAssetBundle(string assetBundleName)
    {
        if (assetBundleMap == null)
        {
            return null;
        }
        List<string> assetPaths;
        if (assetBundleMap.assetBundleDict.TryGetValue(assetBundleName, out assetPaths))
        {
            assetPaths.Sort(EditorUtility.NaturalCompare);
            return assetPaths;
        }
        return null;
    }

    /// <summary>
    /// 最终需要进行打包的资产构建列表
    /// </summary>
    /// <returns></returns>
    public static AssetBundleBuild[] GetAssetBundleBuilds()
    {
        List<AssetBundleBuild> assetBundleBuilds = new List<AssetBundleBuild>();
        string tempPath = "Temp/AssetBundleBuildAnalyze";
        if (assetBundleMap == null)
        {
            assetBundleMap = AssetBundleMap.MakeByLoad(tempPath);
        }
        else
        {
            assetBundleMap.Save(tempPath);
        }
        assetBundleBuilds.AddRange(assetBundleMap.GetBuilds());

        return assetBundleBuilds.ToArray();
    }

    private class AssetBundleMap
    {
        public readonly HashSet<string> assetPathSet = new HashSet<string>();
        public Dictionary<string, List<string>> assetBundleDict = new Dictionary<string, List<string>>();

        public void AddMap(string assetPath, string assetBundleName)
        {
            if (string.IsNullOrEmpty(assetPath) || string.IsNullOrEmpty(assetBundleName))
            {
                return;
            }
            if (assetPathSet.Contains(assetPath))
            {
                return;
            }

            assetPathSet.Add(assetPath);
            List<string> assetPaths;
            if (assetBundleDict.TryGetValue(assetBundleName, out assetPaths))
            {
                assetPaths.Add(assetPath);
            }
            else
            {
                assetPaths = new List<string> { assetPath };
                assetBundleDict.Add(assetBundleName, assetPaths);
            }
        }

        public List<AssetBundleBuild> GetBuilds()
        {
            List<AssetBundleBuild> assetBundleBuilds = new List<AssetBundleBuild>();
            foreach (var kv in assetBundleDict)
            {
                assetBundleBuilds.Add(new AssetBundleBuild() { assetBundleName = kv.Key, assetNames = kv.Value.ToArray() });
            }
            return assetBundleBuilds;
        }

        public void Save(string tempPath)
        {
            using (MemoryStream stream = new MemoryStream())
            {
                new BinaryFormatter().Serialize(stream, assetBundleDict);
                File.WriteAllBytes(tempPath, stream.ToArray());
            }
        }

        public static AssetBundleMap MakeByLoad(string tempPath)
        {
            AssetBundleMap map = new AssetBundleMap();
            if (File.Exists(tempPath))
            {
                byte[] bytes = File.ReadAllBytes(tempPath);
                using (MemoryStream stream = new MemoryStream(bytes))
                {
                    map.assetBundleDict = new BinaryFormatter().Deserialize(stream) as Dictionary<string, List<string>>;
                }
            }
            return map;
        }
    }
}

最终调用的方式如下:

private static void BuildAssetBundlesInter(string outputPath, AssetBundleBuildOptions options)
{
    AssetBundleBuild[] assetBundleBuilds = AssetBundleBuildMap.GetAssetBundleBuilds();

    AssetBundleManifest manifest = BuildPipeline.BuildAssetBundles(outputPath, assetBundleBuilds,
            BuildAssetBundleOptions.UncompressedAssetBundle, GetBuildTarget());
    AssetBundleBuildMap.Clear();
}
### 回答1: Unity AssetBundle Browser是Unity引擎的一款插件,它可以帮助游戏开发者在编辑器内浏览和管理AssetBundle资源。AssetBundle资源是游戏制作过程中必需的一种资源,它可以将一些预制件、贴图、声音和代码等打成二进制文件,并在运行时通过代码加载到游戏中,从而实现动态加载资源和减少加载时间的效果。 Unity AssetBundle Browser插件提供了一个可视化的资源管理界面,用户可以通过它对资源进行打、解、浏览、删除等操作。在使用AssetBundle资源时,由于资源文件的二进制格式与编辑器内资源格式不同,开发者需要将资源文件打成AssetBundle进行加载。Unity AssetBundle Browser将这个过程简化了,用户只需选择需要打的资源,设置相关参数,并点击打按钮,就可以轻松地创建AssetBundle文件,并将其加载到游戏中。 此外,Unity AssetBundle Browser还支持实时预览AssetBundle资源,开发者可以通过它快速地浏览预制件、纹理、材质球等资源,从而提高开发效率。总的来说,Unity AssetBundle Browser是一款十分实用的插件,它为开发者提供了一种便捷的方式去管理和使用AssetBundle资源,使游戏制作变得更加高效和便捷。 ### 回答2: Unity AssetBundle Browser是一个Unity扩展工具,用于在IDE中浏览和管理AssetBundle。AssetBundle是一种Unity资源,可以使开发者轻松创建和管理资源,如纹理、声音和模型等。 Unity AssetBundle Browser可以让开发者直观地查看AssetBundle文件的内容,括资源列表、依赖关系和预览功能。此外,它还提供了打、解、压缩和解压缩AssetBundle文件的功能,使开发者可以更灵活地使用AssetBundle,以满足项目需求。 使用Unity AssetBundle Browser,开发者可以快速地浏览和编辑AssetBundle,节省了大量的时间和精力。它还可以帮助开发者更好地理解AssetBundle的工作原理,为开发者深入学习Unity提供了一个重要的工具。 总之,Unity AssetBundle Browser是一个非常有用的工具,可以帮助Unity开发者更高效地管理和使用AssetBundle,提高项目开发效率和质量。 ### 回答3: Unity AssetBundle Browser是Unity游戏引擎中的一个工具,可以用于浏览和导出AssetBundle资源。AssetBundle是一种Unity的资源打方式,可以将游戏中的资源打成一个文件,方便后续的版本管理和部署。 使用AssetBundle Browser可以很方便地管理和处理AssetBundle资源括在游戏中动态加载和卸载资源,减小游戏的内存占用和优化游戏性能。通过AssetBundle Browser,我们可以查看和编辑资源中的资源,例如模型、贴图、音频等,并提供了一套完整的编辑工具,支持资源导入、复制、删改等操作。 在游戏开发中,AssetBundle Browser可以用于优化游戏的资源处理方式,提高游戏性能,同时也可以帮助开发者更加高效地管理和打游戏资源。值得注意的是,AssetBundle Browser只适用于Unity游戏引擎,需要Unity开发环境的支持。因此,开发者在使用AssetBundle Browser时,需要提前了解Unity的相关知识和技术,否则会影响到开发效率和质量。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值