Unity构建详解(4)——SBP的依赖后处理

【AddHashToBundleNameTask】

这个Task的作用很明确,给Bundle的名字附加一个Hash值:

  • 根据bundle所包含的asset计算出来一个hash值添加在原来Bundle的末尾
  • 替换了BundleBuildContent.BundleLayout和aa.bundleToAssetGroup中的key

有两个问题,为什么要计算Hash值,以及如何计算。

Hash有两个作用:一是数据压缩;二是数据加密。这里是为了给Bundle的名字加密。

Hash值计算流程:

  • 从BundleBuildContent.BundleLayout中得到bundleName和及该Bundle包含的Asset
    • SBP有一个配置界面,可以配置某个Bundle的Name和哪些Asset同属于某个Bundle
  • 计算该asset的依赖关系得到一个Hash128
    • AssetDatabase.GetAssetDependencyHash来计算

    • 如果有缓存,可以直接从缓存中读取

  • 用所有Asset的Hash128算出来一个RawHash

    •  HashingMethods.Calculate(hashes.ToArray()

 Hash计算算法:

默认的Hash算法是MD5,用的C#的MD5CryptoServiceProvider

另一种是SpookyHash,一种比较新的Hash算法,Unity按照算法原理自己实现的

Hash算法接受的输入是byte[],通常我们需要将输入数据转为byte[],转化的实现看HashingMethods.GetRawBytes。

这里主要是因为要根据依赖计算Hash值,所以才放在后面,如果仅仅根据指定的Bundle的名字来算Hash值,可以去掉这个Task。

Hash算法

hash算法不用自己去实现,知道在什么样的情况下用什么样的Hash算法即可:

  • 获取一个文件、图片等的Hash值
    • 使用MD5或SHA系列
  • 获取各种基元数据组合起来的Hash值,例如获取一个类的hash值,或者一系列类的hash值
    • 使用MD5或SHA系列
  • 获取几个int类型的数组组合的Hash值
    • 用简单的乘法Hash,例如计算3个数的hash值,hash = hash *n + x1,hash = hash *n+x2,hash = hash*n +x3。为防止产生hash冲突。一般n取素数,例如31,131,397等,可以每次计算时用不同的n,也可以用相同的n
    • 如果是float和int的混合,先将float乘以100,1000等换算成int
  • 获取一个或多个短字符串的hash值
    • 使用FNV、MurMurHash、FramHash、SpookyHash、xxhash等
    • 内置的String.GetHashCode()

【StripUnusedSpriteSources】

  • 从SpriteData中找到所有需要打包的(PackedSprite)的Sprites的SourceTexture
  • 拿到所有引用的Object
    • 从DependencyData.AssetInfo拿到所有Asset引用的Object
    • 从DependencyData.SceneInfo拿到到所有Scene引用的Object
  • 两者可以选出没被引用的SourceTexture
  • DependencyData.AssetInfo中找到没被引用的sprite对应的SourceTexture的includedObjects,并剔除第一个object,也即sprite对应的object
  • 如果ExtendedAssetData也包含该asset,一样剔除

这里Task存在的主要原因是:指定哪些Asset要被打包时不是一个个指定的,可以指定贴图所在的文件夹,此时文件夹内的贴图都被认为是要打包的Aseert,但实际上可能有些贴图根本不需要被打包,这里就是为了去掉不需要使用的贴图。

【CreateBuiltInShadersBundle】

  • 内置的资源有一个独特的guid
    • UnityBuiltInExtraGuid = "0000000000000000f000000000000000"
  • 从DependencyData.AssetInfo的所有引用的Object中寻找guid等于该内置的guid的Object
  • 从DependencyData.SceneInfo的所有引用的Object中寻找guid等于该内置的guid的Object
  • 获取该资源的类型BuildCacheUtility.GetMainTypeForObjects
    • 依赖计算时有获取Object包含的Type,第一个Type就是MainType
  • 从这些类型中找到资源类型是Shader的Obejct,添加进BundleExplictObjectLayout中
    • m_Layout.ExplicitObjectLocation.Add(usedSet[i], ShaderBundleName)
    • ShaderBundleName可以自己指定

【CreateMonoScriptBundle】

处理流程类同Shader,只是资源类型变成MonoScript,ShaderBundleName变成MonoScriptBundleName。

我们写的每个脚本都会有一个对应的MonoScript,MonoScript.FromMonoBehaviour这个函数能够通过MonoBehaviour对象,找到MonoScript对象。反之,通过MonoScript对象能够找到对应的脚本类。

因此,bundle内不会包含脚本内容数据,有的是MonoScript,进而找到对应的类。

  • 13
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值