一个灵活的AssetBundle打包工具

尼尔:机械纪元

上周介绍了Unity项目中的资源配置,今天和大家分享一个AssetBundle打包工具。相信从事Unity开发或多或少都了解过AssetBundle,但简单的接口以及众多的细碎问题也给工作带来较多的困扰。今天分享AssetBundle工具的实践与想法,相信这块内容对帮助理解AssetBundle有较大的帮助。

Unity提供了两种资源加载方式,一种是Resources,另外种就是AssetBundle。所有的资源只要放在Resources目录下,在打包的时候会自动打进去,并可以通过相应的接口加载。正常情况下Resources非常方便,可以满足日常的需求,但资源放Resources会带来资源更新上的问题。之前写过一篇文章Unity资源目录及加载接口介绍可以了解些细节。

假设首包所有资源都放Resources,后续更新资源的走AssetBundle,会发现AssetBundle和Resources的资源互相不兼容。当调整一个模型的材质参数后,对模型进行打包仍需要把Mesh,Texture等资源都打进去。这会导致更新包过大,同时在加载这个模型时,这些资源是不共用的,相同的资源可能在内存中存在两份。所以正常情况下,项目发布时所有需要更新的资源要打成AssetBundle。

正常项目中资源的提交与变更非常频繁,手工对每个资源配置Bundle费时费力,基本不可取。所以一般项目中的Bundle都是程序自动创建的。同时为了避免有多余的资源被打包,通常需要配置哪些资源是发布资源(直接加载的),其他资源通过引用的形式获取。这个配置需要方便修改,来满足日常变更。

Bundle的打包规则对资源加载速度,更新大小,重复资源数量以及最终包数量等等都有较大影响。一个可靠的Bundle打包方案应该是根据实际情况对Bundle打包规则做调整慢慢产生的。

在Unity 4,只有最基础的几个打包接口可以用于打包。Unity 5简化了Bundle打包时候的依赖关系,但实际如何创建Bundle以及对依赖资源的配置都节省不了。远远不能满足项目对资源打包这块的需求。

这里实现的AssetBundle打包工具帮助简化这个繁琐的打包过程,同时方便做规则调整,得到更优的打包方案。目前工具BundleBuildTool已经放在GitHub,可以作为一份打包实现的参考,也可以直接使用这工具来进行打包。

AssetBundle

An AssetBundle is an archive file containing platform specific Assets (Models, Textures, Prefabs, Audio clips, and even entire Scenes) that can be loaded at runtime.

资源类型

不同类型资源会有不同的打包方式,比如场景文件的打包接口和其他资源的打包接口就是不一样的。通过定义不同的资源类型,可以实现不同的打包方式,支持更多资源的打包。

public enum BundleType
{
    None = 0,
    Script,         // .cs
    Shader,         // .shader or build-in shader with name
    Font,           // .ttf
    Texture,        // .tga, .png, .jpg, .tif, .psd, .exr
    Material,       // .mat
    Animation,      // .anim
    Controller,     // .controller
    FBX,            // .fbx
    TextAsset,      // .txt, .bytes
    Prefab,         // .prefab
    UnityMap,       // .unity
}

对于特殊类型的资源,通过类型可以做一些定制化操作。比如把所有的Script配置在一个Bundle里面,然后在启动的时候对这个Bundle做预加载。通常情况下也会把所有的Shader配置到一个Bundle里面。

正常一个模型会有自己的Texture,Mesh & Animation,把资源按类型打成三个包,在加载的时候可以得到更高的加载速度。Unity异步加载接口会同时进行多个资源加载,资源配置在不同的包里,可以有较好的加载速度提升,所以一般是按资源类型来进行打包。不过要注意如果太分散的话,一样会影响加载速度。

资源加载速度这个是在文章Asset Bundles vs. Resources: A Memory Showdown提及。

These blocks sizes are optimized for loading multiple Assets and bundles in parallel. For example, you should be able to load objects from 4 to 5 Asset Bundles at the same time without the the allocators for Asset Bundle Async loading or Type Trees needing new blocks.

资源依赖

处理资源依赖应该是打包过程最复杂的一块功能,这里把获取资源依赖文件列表单独设计一个类,做一些特殊情况处理。如果发现一些依赖关系上的错误,除了修改资源本身外,也可以在打包环节实现一些脚步做保障。

正常情况下,通过AssetDatabase.GetDependencies即可获取一个资源的所以依赖文件。但实际情况中,Unity内部是通过分析内部guid来生成依赖文件。有时候在文件里面会存在一些脏的guid这会产生多余的依赖。比如你修改一个材质贴图属性名,然后设置了一张新的贴图给这个新的属性名。打开材质文件会发现旧的属性名以及引用guid出现在材质文件,通过GetDependencies获取的最后结果也包含这个数据。实现自己获取依赖函数来处理这种多余依赖关系。同时提供带缓存接口,提高打包效率。

下面是对材质依赖贴图文件获取的代码实现。

...
MaterialProperty[] proTes = MaterialEditor.GetMaterialProperties(new Object[] {mat});
for (int i = 0; proTes != null && i < proTes.Length; ++i)
{
    if (proTes[i].type == MaterialProperty.PropType.Texture)
    {
        Texture tex = mat.GetTexture(proTes[i].name);
        string path = AssetDatabase.GetAssetPath(tex);
        if (!dict.ContainsKey(path))
        {
            dict.Add(path, path);
        }
        Resources.UnloadAsset(tex);
    }
}   
...

资源剔除

处理完资源依赖后,还碰到一个问题就是最后打包Assets资源。通过AssetDatabase.LoadAllAssetAtPath获取这个文件依赖的所有的Assets资源。如果对所有的这些Assets资源都做打包的话,会发现一些编辑器用数据也会被打包进去。特别是对于FBX类型文件,通常会存在一个"__preview_Take 001"的动作资源使包体变大很多。对于这些不必要的数据,在打包环节中增加一个剔除规则,减少包体大小。

public static List<UnityEngine.Object> FilterObjectByType(UnityEngine.Object[] assets, BundleType bundleType)
{
    List<UnityEngine.Object> ret = new List<UnityEngine.Object>();
    foreach (UnityEngine.Object asset in assets)
    {
        switch (bundleType)
        {
        case BundleType.FBX:
            if (!(asset.GetType() == typeof(AnimationClip) && asset.name == "__preview_Take 001"))
            {
                ret.Add(asset);
            }
            break;
        default:
            ret.Add(asset);
            break;
        }
    }
    return ret;
}

Unity 5刚出的时候会把这个数据打进AssetBundle造成包体过大,后面版本观察已经修复这个问题。不过也可以发现这个环节的必要性,如果发现资源出问题在这个环节处理即可。

这个环节不仅可以剔除不必要的数据,还可以直接修改数据本身。就拿Mesh数据举例,美术在制作过程中会导出多余的顶点数据在文件里面(uv3,uv4...)。通常配置Optimize Mesh可以干掉这些无用数据,不过直接启用可能会出现删除了需要数据情况,比如color数据丢失。所以自己来做,通过把Mesh对象上不需要的对象数据置空,然后再打包即可。在之前分享的资源配置工具里已经做了对Mesh顶点数据的配置,基本上就是为这个打包环节服务,因为无法修改FBX文件,只能美术重新导出。

资源大小

资源大小影响最后的包体大小,如果对包体大小以及更新量有关注的话,对资源大小做预估是一个非常有必要的环节。在资源大小计算环节,不能疏漏之前二个资源环节对资源的处理,同时不同类型的资源统计方式不一样。

通常通过下面两个方式预估资源大小

int resSize = UnityEngine.Profiling.Profiler.GetRuntimeMemorySize(asset);
FileInfo fileInfo = new FileInfo(assetPath);
int fileSize = fileInfo.Length;

如何对一个资源做一个大小估算,并不是一件非常方便的事情的。如果依赖资源已经在之前打包了,那这个资源的实际大小是要考虑减去依赖资源那部分的大小。如果不统计依赖资源的大小,那这个资源的包的大小也是不准确的。所以这里的实际逻辑较为复杂,但实际一个大致的值就可以了,然后观察最后的包大小做一些配置微调即可。

Bundle模型

讨论完资源上的一些细节,下面开始Bundle设计的介绍。一个Bundle模型用name做唯一标识,为了方便管理加入了parent与children数据。同时一个Bundle应该有一个固定资源类型。为了方便对包大小做限制加入了size属性,作为资源大小的预估。

public class BundleData
{
    public string name = string.Empty;
    public string parent = string.Empty;
    public BundleType type = BundleType.None;
    public BundleLoadState loadState = BundleLoadState.UnloadImmediately;
    public int size = 0;
    public List<string> includs = new List<string>();
    public List<string> children = new List<string>();
}

最后一个Bundle包含多个资源文件路径。尽管AssetBundle是按Assets打包的,但在正常环境下的资源是以文件存在的。一个资源文件可能包含多个资源,也可能引用到其他资源。资源文件可以用路径来标识,Unity内部通过GUID来标识资源文件,所以即使你挪动文件因为GUID不变,还是可以找到这个文件。这里决定直接用资源路径来标识资源而不是使用GUID,因为挪动资源目录有较多的风险,原则上禁止挪动资源。如果真挪动了资源,按最新的资源路径生成Bundle是一个不错的选择。

如果有对Bundle有其他属性上的需求,在这个类扩展就好。

Bundle创建规则

定义Bundle后,创建Bundle是很困扰的一个问题。在大型项目中,资源的量非常大,资源之间的互相引用也较为复杂。这里定义一个数据结构帮忙创建Bundle。

public class BundleImportData
{
    public string RootPath = "";
    public string FileNameMatch = "*.*";
    public int Index = -1;
    public int TotalCount = 0;
    public BundleType Type = BundleType.None;
    public BundleLoadState LoadState = BundleLoadState.OnUnloadAsset;
    public bool Publish = false;
    public int LimitCount = -1;
    public int LimitKBSize = -1;
    public bool PushDependice = false;
    public bool SkipData = false;
}

对于一个Bundle,可以约束它的大小,对象数量、类型、加载方式、打包方式。然后根据规则,自动给每个资源文件配置Bundle。

资源分加载资源和被依赖引用到的资源,对于直接加载的资源,需要配置Publish为True。Bundle创建就是从这些配置了Publish的资源文件以及其依赖生成的。

对所有可能被打包的资源配置打包规则,没有被配置资源文件,则会被一起打倒最后资源的包里面。这里会碰到一个问题,有些资源需要补分包,但是通用规则会包含不需要分包的资源。这里增加了一个SkipData属性,当为True时这些资源不单独创建Bundle。

然后讨论下PushDependice属性,正常情况下只有在打Prefab类型的资源的时候才会做这个操作。因为Prefab数据本身是不共享的,然后避免Prefab与Prefab之间的复杂依赖。

最后讨论下打包的顺序,因为资源之间有互相依赖,所以需要配置资源的打包顺序。这里资源的打包顺序就是BundleImportData创建的顺序。这里需要对资源之间的依赖以及资源类型有一定的认识。

已经配置过Bundle的资源不会变更,新增的资源会按规则配置相应的Bundle。通常规则发生变更会影响非常多的资源,如果所有资源重新配置会导致更新包过大。

Bundle构建

首次创建的Bundle,由于本地文件不存在,会触发构建。然后资源之间有互相依赖,所有被依赖的Bundle也需要参加构建。对于增量构建,这里做了一个简化设计,不自己去计算文件是否变更,而是由外部提供一个文件变化列表。通过这个列表工具自动生成Bundle构建列表,提高打包速度。

在配置打包参数为BuildAssetBundleOptions.DeterministicAssetBundle后,如果不对资源做修改,两次打包的文件是一样的。所以即使有很多资源因为依赖要重新打包,最后的文件未发生变化,就不会触发更新。

Bundle索引

Bundle构建完后只是一堆二进制文件,需要根据Bundle之间的依赖关系生成出一份数据。除了需要知道Bundle之间的依赖之外,同时还需要知道资源路径与Bundle之间的映射关系。最后还要把Bundle状态信息保存下来,用于Bundle更新、加载和卸载。

public class BundleState
{
    public string bundleID = string.Empty;
    public uint crc = 0;
    public uint compressCrc = 0;
    public int version = -1;
    public long size = -1;
    public BundleLoadState loadState = BundleLoadState.OnUnloadAsset;
    public BundleStorePos storePos = BundleStorePos.Building;
}
// like UnityEngine.AssetBundleManifest
public class BundleManifest { ... }

这个文件自己定义形式,可以使分散的多个文件,也可以统一放到一个文件里面,自己实现可以优化数据结构减少内存开销。

通用的Bundle打包方案

下面是在Unity Standard Assets资源上做配置后的结果

BundleBuildTool

按大小配置基础资源,然后对于Prefab和Unity文件限定下个数,避免过多的资源依赖。配置结束后点击CreateBundle就可以得到下面的结果。

[完 2017-07-13 Carber]

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容