YooAsset插件
学习YooAsset插件和分析
YooAsset是一个Unity资源系统,集合资源打包、分包、更新、加载等。
信息
- 官网:YooAsset
- 版本:1.4.2
- 源码:https://github.com/tuyoogame/YooAsset
构建
收集
- 检测配置合法性
- 收集所有Package的资源(遍历所有的Group,然后再遍历所有的Collector)
AssetBundleCollectorSettingData.Setting.GetPackageAssets(buildMode, packageName);
- 剔除未被引用的依赖资源
- 录入所有收集器收集的资源
- 录入相关依赖的资源
- 记录关键信息
- 填充主动收集资源的依赖列表
- 计算完整的资源包名
- 移除不参与构建的资源
- 构建资源包
Runtime
初始化
- YooAssets初始化。
- 创建驱动器
YooAssetsDriver
用于调度系统的Update
- 创建远程调试器
RemoteDebuggerInRuntime
用于调试加载情况 - 初始化异步系统
OperationSystem
- 创建驱动器
// 初始化资源系统
YooAssets.Initialize();
- 创建Package。
- 创建package
- 设置为默认的package
// 创建默认的资源包
var package = YooAssets.CreateAssetsPackage("DefaultPackage");
// 设置该资源包为默认的资源包,可以使用YooAssets相关加载接口加载该资源包内容。
YooAssets.SetDefaultAssetsPackage(package);
- 初始化资源系统(Editor模拟)
EditorSimulateModeHelper.SimulateBuild
会调用AssetBundleSimulateBuilder
构建资源清单- 再把
Parameters
传入package
进行资源清单初始化 package
根据mode
创建EditorSimulateModeImpl
,这是包含了IPlayModeServices
和IBundleServices
具体实现的集合。package
创建AssetSystemImpl
,然后进行初始化。EditorSimulateModeImpl
进行初始化。- 调用
EditorSimulateModeInitializationOperation
加载清单,并且设置清单到EditorSimulateModeImpl
var createParameters = new EditorSimulateModeParameters();
createParameters.SimulatePatchManifestPath = EditorSimulateModeHelper.SimulateBuild(packageName);
InitializationOperation initializationOperation = package.InitializeAsync(createParameters);
- 初始化资源系统(单机模式)
- 把
Parameters
传入package
进行资源清单初始化 package
根据mode
创建OfflinePlayModeImpl
,这是包含了IPlayModeServices
和IBundleServices
具体实现的集合。package
创建AssetSystemImpl
,然后进行初始化。OfflinePlayModeImpl
进行初始化。- 调用
OfflinePlayModeInitializationOperation
- 获取Package版本号
- 加载清单。
- 设置清单到
OfflinePlayModeImpl
- 校验清单的资源包,将正确的补丁放入
CacheSystem
- 把
var createParameters = new OfflinePlayModeParameters();
InitializationOperation initializationOperation = package.InitializeAsync(createParameters);
- 初始化资源系统(联机模式)
- 把
Parameters
传入package
进行资源清单初始化 package
根据mode
创建HostPlayModeImpl
,这是包含了IPlayModeServices
和IBundleServices
具体实现的集合。package
创建AssetSystemImpl
,然后进行初始化。HostPlayModeImpl
进行初始化。- 调用
HostPlayModeInitializationOperation
- 获取补丁水印(安装包版本号和补丁版本号),如果水印不一致,则清空缓存目录。
- 获取缓存目录Package版本号,如果存在,则加载和解析缓存目录清单
- 在缓存目录清单加载失败(版本号不对、hash值不对、清单不存在等),则获取安装包Package版本号,并且把版本号和清单写入到缓存目录。最后再加载安卓包清单。
- 设置清单到
HostPlayModeImpl
- 校验清单的资源包,将正确的补丁放入
CacheSystem
- 把
var createParameters = new OfflinePlayModeParameters();
InitializationOperation initializationOperation = package.InitializeAsync(createParameters);
更新
//更新Package版本信息
var operation = package.UpdatePackageVersionAsync();
yield return operation;
//更新Package资源清单
package.UpdatePackageManifestAsync(operation.PackageVersion);
//创建下载器
var downloader = YooAssets.CreatePatchDownloader(downloadingMaxNum, failedTryAgain);
只有
联机模式
才有更新系统
HostPlayModeUpdatePackageVersionOperation
请求远端版本号信息。HostPlayModeUpdatePackageManifestOperation
更新补丁清单。- 加载
Cache
清单,判断版本号是否为最新 - 当版本号不是最新时,下载远端清单到
Cache
目录。 - 重新加载
Cache
清单。 - 校验清单的资源包,将正确的补丁放入
CacheSystem
- 加载
PatchDownloaderOperation
下载补丁(不包含已有缓存、内置资源)- 清除过期的补丁文件
扩展
由于代码比较简单易懂,并且框架解耦的较好,扩展性比较好。
更新
目前热更方式为CDN获取版本信息,可以增加后台模式,以适应后台更新的方案。
//AssetsPackage.InitializeAsync()
if (...)
{
...
}
else if (_playMode == EPlayMode.ServerPlayMode)
{
var serverPlayModeImpl = new ServerPlayModeImpl();
...
}
疑问
- 覆盖安装情况下,资源是否正常?
会通过水印判断安装包是否有变更,如果安装包变更了,则清除
Cache
目录
- 断点续传是否会出现缺漏的问题?
每次会对文件计算hash值进行校验。没有缺漏问题。
- 热更是否要在Cache增加一份副本?
安装包的内容只有一份。热更时会使用
IsBuildinPatchBundle
进行判断剔除。
- CDN需要全量补丁?
发布没有差异补丁的概念,所以都是全量补丁
对比
Addressable
- Group与Addressable的差不多,都是用于资源分组使用
- 没有了自定义Entry的概念,取而代之增加了Collector对资源进行收集设置。
- 收集器的地址保存的是字符串,移动文件夹时会出现丢失的问题。
- 获取Entry地址和寻址信息需要手动拼写
- Tags只能对Group设置,无法只对Collector和Entry设置
- Group相对于Addressable去除了很多独立设置,无法对AB进行独立设置压缩方式,命名方式等
- 地址和寻址两套方式可以选择使用
- 支持旧的构建模式和SBP模式
- 配置有Json和Bytes两种,bytes只是简单序列化了,并没有做压缩处理。
- Group之上还增加了Package的概念,用于构建时可以分情况构建。
- 有多种构建模式,例如演练和模拟模式可以方便调试
- 有调试器和报告查看工具
- 更新地址需要运行设置。
- 支持构建原文件
- Editor模拟模式,需要每次运行时调用构建,需要生成列表清单。目前只有收集到的资源需要生成清单,理论上速度应该不慢。
- 构建时才会检查资源是否重复收集
- Operation采用的是Update来处理,没有采用协程。Operation通过状态来去区分任务,需要等待子Operation完成来改变状态。
- 有加密接口。Android采用加密只能使用偏移头文件的方式,如果采用其他方式,将无法加载SteamingAsset目录资源。
问题
- 在AssetBundleFileLoader加载失败时会调用CacheSystem.DiscardFile清除,但实际上会清除整个Cache目录!!!
- 缺少SpriteAtlas打包
- 加载场景时,会自动清空资源引用计数,导致对象池中的对象丢失资源引用。
- 只有在加载场景时才会检测资源引用计数
优点
- 代码结构较为简单,容易扩展
- 多线程下载
- 多线程校验文件
- 开箱即用
缺点
- 每次都需要校验文件,全量计算文件Hash,数量一多会比较耗时
- 收集器需要填写配置,有资源需要特殊处理时,比较难处理
- 补丁为全量资源上传CDN,如果改成增量补丁,并且按版本分开目录,需要扩展新的mode。
- 寻址的Key值无法自定义,相同文件名会出现Key重复的问题。
评论
还沒有留言。