网易首页 > 网易号 > 正文 申请入驻

Unity社区精华帖 | SBP - Scriptable Build Pipeline

0
分享至

Scriptable Build Pipeline 是什么?能来带什么好处?

自定义 Unity 如何构建应用内容

将原先处于C++的引擎代码移到了C#

加速了AssetBundle的构建时间

改善增量构建处理

对开发者来说具备更多的灵活性

Unity 版本要求:Unity 2018.3 +

术语

Asset

磁盘上的源文件,通常位于项目的 “Assets” 文件夹中。此文件在内部导入到您的资产的可用于游戏的表示形式,其中可以包含多个 Object(SubAsset)。

Object (SubAsset)

单个 Unity 可序列化单元。也称为 SubAsset。导入的 Asset 由一个或多个对象组成。

Includes

从中构造 Asset 的唯一一组 Object。

References

一组需要被 Asset 使用,但不在该 Asset 中的 Object。

快速入门 & BuildIn切换

BuildAssetBundles

BuildIn Pipeline


using System.IO;using UnityEditor;
public static class BuildAssetBundlesExamplepublic static bool BuildAssetBundles(string outputPath, bool forceRebuild, bool useChunkBasedCompression, BuildTarget buildTarget)var options = BuildAssetBundleOptions.None;if (useChunkBasedCompression)options |= BuildAssetBundleOptions.ChunkBasedCompression;
if (forceRebuild)options |= BuildAssetBundleOptions.ForceRebuildAssetBundle;
Directory.CreateDirectory(outputPath);var manifest = BuildPipeline.BuildAssetBundles(outputPath, options, buildTarget);return manifest != null;

SBP

CompatibilityBuildPipeline 是 SBP 目前官方使用 SBP 流程适配 BuildIn 流程的适配实现。

SBP 不支持以前支持的所有功能。


using System.IO;using UnityEditor;// Added new usingusing UnityEditor.Build.Pipeline;
public static class BuildAssetBundlesExamplepublic static bool BuildAssetBundles(string outputPath, bool forceRebuild, bool useChunkBasedCompression, BuildTarget buildTarget)var options = BuildAssetBundleOptions.None;if (useChunkBasedCompression)options |= BuildAssetBundleOptions.ChunkBasedCompression;
if (forceRebuild)options |= BuildAssetBundleOptions.ForceRebuildAssetBundle;
Directory.CreateDirectory(outputPath);// Replaced BuildPipeline.BuildAssetBundles with CompatibilityBuildPipeline.BuildAssetBundles herevar manifest = CompatibilityBuildPipeline.BuildAssetBundles(outputPath, options, buildTarget);return manifest != null;

注意事项

CompatibilityBuildPipeline.BuildAssetBundles vs BuildPipeline.BuildAssetBundle 的特性与行为的对比。

AssetBundleBuild.addressableNames:

https://docs.unity3d.com/ScriptReference/AssetBundleBuild-addressableNames.html

BuildAssetBundleOptions

AssetBundle SBP 构建流程

01 Setup - 平台环境初始化

SwitchToBuildPlatform

RebuildSpriteAtlasCache

02 Player Scripts - 工程源代码编译

BuildPlayerScripts、PostScriptsCallback

03 Dependency

CalculateSceneDependencyData

CalculateCustomDependencyData (UNITY_2019_3_OR_NEWER)

CalculateAssetDependencyData

StripUnusedSpriteSources

PostDependencyCallback

04 Packing

GenerateBundlePacking

GenerateBundleCommands

GenerateSubAssetPathMaps

GenerateBundleMaps

PostPackingCallback

05Writing

WriteSerializedFiles

ArchiveAndCompressBundles

AppendBundleHash

GenerateLinkXml

PostWritingCallback

06 Generate manifest files

官方 TODO 项

01 Setup

SwitchToBuildPlatform

切换当前平台为构建 AssetBundle 的目标平台对应的 BuildTargetGroup 、 BuildTarget。

执行回调 IPreprocessShaders 、IProcessScene、IProcessSceneWithReport

关键逻辑:


EditorUserBuildSettings.SwitchActiveBuildTarget(m_Parameters.Group, m_Parameters.Target);

RebuildSpriteAtlasCache

针对目标平台重新构建 工程内所有的 SpriteAtlas。

关键逻辑:


SpriteAtlasUtility.PackAllAtlases(m_Parameters.Target);

02 Player Scripts

BuildPlayerScripts

编译目标平台源代码生成 ScriptInfo(TypeDB - 记录了脚本类型 、 属性数据)

TypeDB 为后续的序列化提供正确的字段名称。

关键逻辑:


PlayerBuildInterface.CompilePlayerScripts(m_Parameters.GetScriptCompilationSettings(), m_Parameters.ScriptOutputFolder);

PostScriptsCallback

回调脚本订阅了脚本编译完成节点的所有函数。

上层开发者,注册监听 BuildCallbacks.PostScriptsCallbacks。

关键逻辑:


IScriptsCallback.PostScripts(IBuildParameters parameters, IBuildResults results);

03 Dependency

CalculateSceneDependencyData - 计算所有场景的依赖信息

判断是否使用 Cache 机制;计算场景相关依赖项

关键逻辑:


#if UNITY_2019_3_OR_NEWERContentBuildInterface.CalculatePlayerDependenciesForScene(scenePath, settings, usageTags, m_DependencyData.DependencyUsageCache);#elseContentBuildInterface.PrepareScene(scenePath, settings, usageTags, m_DependencyData.DependencyUsageCache, outputFolder);#endif

Cache 仅记录预制相关的依赖。

CalculateCustomDependencyData (UNITY_2019_3_OR_NEWER)

计算未被 AssetDatabase 追踪的自定义 Assets 依赖的 Object 关系;

由 IBundleBuildContent.CustomAssets 自定义 AssetBundle 包含的内容。

关键逻辑:


foreach (CustomContent info in IBundleBuildContent.CustomAssets)info.Processor(info.Asset, this);

CalculateAssetDependencyData - 计算所有 Asset的依赖数据

基础入参

源代码的 TypeDB

GraphicsSettings

场景的 LightSetting

获取 Asset 的所包含的 Objects

即为该 Asset 引用的 Object;为 该 Asset YAML 序列化中 !u![ClassID] 段所表示的 Object。

通过 TypeDB 获取 Objects 所引用的 Objects

如果是 Sprite 类型资源,则进行 Sprite Atlas 的打包设置

计算 SubAssets

关键逻辑:


var includedObjects = ContentBuildInterface.GetPlayerObjectIdentifiersInAsset(asset, input.Target);var referencedObjects = ContentBuildInterface.GetPlayerDependenciesForObjects(includedObjects, input.Target, input.TypeDB);ObjectIdentifier[] representations = ContentBuildInterface.GetPlayerAssetRepresentations(asset, target);// Main Asset always returns at index 0, we only want representations, so check for greater than 1 lengthif (representations.IsNullOrEmpty() || representations.Length < 2)return;extendedData = new ExtendedAssetData();extendedData.Representations.AddRange(representations.Skip(1));

StripUnusedSpriteSources - 删除 Assets引用中被Packer的Sprite

筛选出已经被 SpritePacker 打包的Sprite,所归属的 SourceTexture;

删除 Assets 中引用的 Sprite信息;

关键逻辑:


var unusedSources = new HashSet();var textures = m_SpriteData.ImporterData.Values.Where(x => x.PackedSprite).Select(x => x.SourceTexture);unusedSources.UnionWith(textures);
// Count refs from assetsvar assetRefs = m_DependencyData.AssetInfo.SelectMany(x => x.Value.referencedObjects);foreach (ObjectIdentifier reference in assetRefs)unusedSources.Remove(reference);
// Count refs from scenesvar sceneRefs = m_DependencyData.SceneInfo.SelectMany(x => x.Value.referencedObjects);foreach (ObjectIdentifier reference in sceneRefs)unusedSources.Remove(reference);
// SetOutputInformationforeach (var source in unusedSources)var assetInfo = m_DependencyData.AssetInfo[source.guid];assetInfo.includedObjects.RemoveAt(0);
ExtendedAssetData extendedData;if (m_ExtendedAssetData != null && m_ExtendedAssetData.ExtendedData.TryGetValue(source.guid, out extendedData))extendedData.Representations.Remove(source);if (extendedData.Representations.Count == 1)m_ExtendedAssetData.ExtendedData.Remove(source.guid);

PostDependencyCallback

在依赖计算完成后的事件通知

上层开发者,注册监听 BuildCallbacks.PostDependencyCallback

关键逻辑:


IDependencyCallback.PostDependency(IBuildParameters parameters, IDependencyData dependencyData);

04Packing - 资源构建

GenerateBundlePacking - 组装AssetBundle以及计算依赖加载列表

Normal AssetBundle

Streamed Scene AssetBundle

ArchiveFileSystem

校验 Object 是否是 Assets 类型 或 自定义Assets类型 | Scene 类型,创建对应类型的 “容器结构”

可通过 ValidationMethods.ValidAssetFake 接管校验逻辑

计算 Asset 所依赖的加载列表

剔除 Unity默认的内置资源引用 - 处理依赖关系中的重复递归依赖 & 打破环形依赖;(若是 Scene Bundle,则剔除 sharedAssets 的重复引用)

关键逻辑:


foreach (var bundle in m_BuildContent.BundleLayout)if (ValidAssetBundle(bundle.Value, customAssets))PackAssetBundle(bundle.Key, bundle.Value, assetToReferences);else if (ValidationMethods.ValidSceneBundle(bundle.Value))PackSceneBundle(bundle.Key, bundle.Value, assetToReferences);
// Remove Default Resources and Includes for Assets assigned to Bundlesforeach (ObjectIdentifier reference in references)if (reference.filePath.Equals(CommonStrings.UnityDefaultResourcePath, StringComparison.OrdinalIgnoreCase))continue;if (dependencyData.AssetInfo.TryGetValue(reference.guid, out AssetLoadInfo referenceInfo))referencedAssets.Add(referenceInfo);continue;referencesPruned.Add(reference);

GenerateBundleCommands - 为需要序列化的AssetBundle创建WriteOperation行为

初始化 AssetBundle MainAsset

创建对应的 BundleCommand (CreateAssetBundleCommand、CreateSceneBundleCommand + CreateSceneDataCommand)

AssetBundleCommand :

1.建立加载 Asset的入参与 Object 的映射关系

2.记录 AssetsBundle 中包含的Assets相关信息(依赖的Object hash、序列化组织形式等)并排序,以此确保每次序列化数据的稳定性

SceneBundleCommand :

1.建立加载 Scene 所包含需要序列化的Object与加载的入参的映射关系

2.记录 SceneBundle 依赖的Object 至 preloadObjects

SceneDataCommand :

1.建立加载Scene 所需要附加的Assets 的映射关系

2.记录 SceneBundle 依赖的Object 至 preloadObjects

关键逻辑:


Dictionarystring> assetToMainFile = new Dictionarystring>();foreach (var pair in m_WriteData.AssetToFiles)assetToMainFile.Add(pair.Key, pair.Value[0]);
foreach (var bundlePair in m_BuildContent.BundleLayout)if (ValidAssetBundle(bundlePair.Value, customAssets))// Use generated internalName here as we could have an empty asset bundle used for raw object storage (See CreateStandardShadersBundle)var internalName = string.Format(CommonStrings.AssetBundleNameFormat, m_PackingMethod.GenerateInternalFileName(bundlePair.Key));CreateAssetBundleCommand(bundlePair.Key, internalName, bundlePair.Value);else if (ValidationMethods.ValidSceneBundle(bundlePair.Value))var firstScene = bundlePair.Value[0];CreateSceneBundleCommand(bundlePair.Key, assetToMainFile[firstScene], firstScene, bundlePair.Value, assetToMainFile);for (int i = 1; i < bundlePair.Value.Count; ++i)var additionalScene = bundlePair.Value[i];CreateSceneDataCommand(assetToMainFile[additionalScene], additionalScene);

Unity 官方微信

第一时间了解Unity引擎动向,学习最新开发技巧

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
撒贝宁被亲友坑惨!曝其堂哥欠债百万,多个部门介入,曝现场照片

撒贝宁被亲友坑惨!曝其堂哥欠债百万,多个部门介入,曝现场照片

裕丰娱间说
2025-02-18 13:04:21
加州资深内科医生看大S之死:几个意想不到的启示

加州资深内科医生看大S之死:几个意想不到的启示

呼吸界
2025-02-18 19:45:51
大罢免遍地烽火,北市多家餐饮店“谢绝政治活动”:盼好好过日子

大罢免遍地烽火,北市多家餐饮店“谢绝政治活动”:盼好好过日子

新时光点滴
2025-02-18 00:03:50
欧洲国家集体抵制哪吒2!希腊高兴坏了,大搞电影活动吸引游客!

欧洲国家集体抵制哪吒2!希腊高兴坏了,大搞电影活动吸引游客!

青青子衿
2025-02-18 05:44:41
背靠背一轮游世界第3高芙爆冷不敌凯斯勒 多哈迪拜均首战出局

背靠背一轮游世界第3高芙爆冷不敌凯斯勒 多哈迪拜均首战出局

直播吧
2025-02-19 02:39:07
马斯克曝光社保造假后,美国就业数据也发现问题!近25年官方299次修正记录显示:极端高估多在大选年,极端低估多在总统上任首年

马斯克曝光社保造假后,美国就业数据也发现问题!近25年官方299次修正记录显示:极端高估多在大选年,极端低估多在总统上任首年

每日经济新闻
2025-02-18 21:10:09
被社会淘汰的60后70后,余生最好出路是:躺平保命和伺候好父母

被社会淘汰的60后70后,余生最好出路是:躺平保命和伺候好父母

小马达情感故事
2025-02-17 20:00:03
欧冠附加赛:拜仁2-1凯尔特人,进16强!戴维斯绝平,金玟哉送礼

欧冠附加赛:拜仁2-1凯尔特人,进16强!戴维斯绝平,金玟哉送礼

侃球熊弟
2025-02-19 05:46:36
实测马斯克口中最聪明的 AI,就这?

实测马斯克口中最聪明的 AI,就这?

AppSo
2025-02-18 20:37:23
俄外交部发言人:北约拒绝乌克兰“入约”不够,必须放弃2008年做出的承诺

俄外交部发言人:北约拒绝乌克兰“入约”不够,必须放弃2008年做出的承诺

环球网资讯
2025-02-18 21:16:14
168票通过,韩国新总统已定?李在明讲出一句话,给中韩关系定调

168票通过,韩国新总统已定?李在明讲出一句话,给中韩关系定调

大白话瞰世界
2025-02-18 09:07:58
湖南男子高速上睡觉开车,华为智驾送他回家,交警回复:罚款200

湖南男子高速上睡觉开车,华为智驾送他回家,交警回复:罚款200

奇思妙想草叶君
2025-02-18 22:44:19
陕西一父亲带女儿自驾游,4个月后罪行被揭发,判刑9年5个月

陕西一父亲带女儿自驾游,4个月后罪行被揭发,判刑9年5个月

灿烂夏天
2025-02-16 23:28:09
俄罗斯最新民意调查:高达70%俄民众支持将特别军事行动坚持到底!

俄罗斯最新民意调查:高达70%俄民众支持将特别军事行动坚持到底!

凯撒谈兵
2025-02-16 20:04:51
弹琴扰民败诉4年后复仇:胜诉者新车被划!肇事者称影响我孩子3代

弹琴扰民败诉4年后复仇:胜诉者新车被划!肇事者称影响我孩子3代

原广工业
2025-02-18 13:47:08
分裂只是时间问题?联合国大胆预测:13亿人口大国或将一分为三!

分裂只是时间问题?联合国大胆预测:13亿人口大国或将一分为三!

半解智士
2025-02-18 10:55:11
一场1-1踢出欧冠神剧情,拜仁后卫读秒绝杀,孔帕尼欣喜若狂

一场1-1踢出欧冠神剧情,拜仁后卫读秒绝杀,孔帕尼欣喜若狂

足球狗说
2025-02-19 06:10:44
实物黄金遭遇史诗级挤兑!大风暴要来了?

实物黄金遭遇史诗级挤兑!大风暴要来了?

宋鸿兵
2025-02-18 18:04:25
曾属史玉柱的巨人大厦将动工,已烂尾28年,土地使用权仅剩18年

曾属史玉柱的巨人大厦将动工,已烂尾28年,土地使用权仅剩18年

红星新闻
2025-02-18 16:26:12
小伙趁舅舅不在家偷偷报复舅妈,事后舅妈态度大转变,让小伙吃不消

小伙趁舅舅不在家偷偷报复舅妈,事后舅妈态度大转变,让小伙吃不消

神奇的锤子
2025-02-14 16:16:34
2025-02-19 06:20:49
Unity incentive-icons
Unity
Unity中国官方帐户
2257文章数 6716关注度
往期回顾 全部

科技要闻

马斯克发布"最聪明AI":号称碾压DeepSeekV3

头条要闻

10元1个螺母被认定为枪支散件 父子被刑拘获分案调查

头条要闻

10元1个螺母被认定为枪支散件 父子被刑拘获分案调查

体育要闻

曾遭遇两年欠薪,国足最新归化球员是他?

娱乐要闻

陈晓与陈妍希宣布离婚:今后各自安好

财经要闻

存款准备金率5%隐形下限能否突破?

汽车要闻

两种电池可选 小米YU7最大续航820km

态度原创

游戏
手机
亲子
本地
公开课

魂师对决:SP朱竹清返场抽取分析!这个月的压力又是陡然剧增!

手机要闻

iPhone 17 据称将不包含传闻中 iPhone 17 Pro 的圆角矩形相机栏

亲子要闻

娃睡不踏实、老哼唧?这个原因家长很容易忽略(不是冷

本地新闻

非遗版春节|新春青岛行,是谁闯入了动漫世界?

公开课

李玫瑾:为什么性格比能力更重要?

无障碍浏览 进入关怀版