赞
踩
热更新是指,你需要为应用程序修改某种资源,或者增加某种资源的时候,不需要新发布一个新的应用程序到应用商店让用户下载并重新安装,只需要联网,然后下载更新的内容即可。比如游戏出了一款新皮肤,推出一个新活动,或者修复某个紧急的小bug,如果这些小事情每次都要用户重新下载应用程序,就特别烦,尤其是如今中国国民级的手游动辄3到5个G。
为此,考虑需要频繁更新的内容就可以分成两个部分:
Unity没有给我们提供官方的脚本热更方案,而对于普通的资源文件的热更,就是AssetsBundle技术。
在说AssetsBundle之前,还要说说Resources,或者Resources这个目录。
Resources是Unity最早提出的资源动态加载的方案。当你需要动态加载某个资源的时候,你需要把对应的资源,比如预置体,材质等放入Resources目录下,然后用Unity提供的API(Resources.Load()
)去动态加载。
这个目录在以前游戏不是很大的情况下很方便很好用。但在项目做大的时候,它的缺点就暴露得很明显了(缺点来自官方文档):
为了解决这些问题,AssetsBundle诞生了。
AssetsBundle是Unity另一套资源管理的方式。它和Resources的相同之处,也是他们最主要的用途就是允许工程动态加载里面的资源。不同之处在于:
第二个不同之处就是Unity资源热更的核心。而手动构建虽然显得更加繁琐,但也给你更多的选择。你可以选择构建AssetsBundle的路径,选择它的名称和更细分的变体(Variants),你还可以构建好AssetsBundle之后,将其存储到服务器上,让用户在需要资源更新的时候下载新的AssetsBundle。
关于AssetsBundle的第二个话题就是它的存储位置。这里涉及到两个重要的目录:StreamingAssetsPath 和 PersistentDataPath。
StreamingAssetsPath 可以通过Application.streamingAssetsPath
获取路径。在Unity编辑器模式下,它是Assets目录下的StreamingAssets目录。在Android平台里,它就是assets目录。
Android管理资源有两种方式,一种是res/raw目录,res目录里面的文件会参与Android的R文件编译,以便你能够通过R.id去访问,这也同样是为什么你在解压apk,然后尝试读取res/AndroidMenifest.xml时,得到的却是一个二进制文件。另一种就是assets目录。Android对它不会做任何事情,然而,你无法通过文件系统去访问这个目录下的资源。取而代之的是使用Android提供的API——AssetsManager。
和Android一样,Unity也不会对这个目录下的文件做任何事情,包括C#脚本,Shader和材质也不会参与编译(Unity编辑器里,你如果在这个目录下创建一个C#脚本,你会发现它的图标和正常的C#脚本图标不一样,因为Unity根本没把它当成一个需要正常编译的C#脚本)。
你应该使用这个目录去存储Unity的资源。在PC上,你可以直接使用文件系统访问Application.streamingAssetsPath
获取里面的资源。而在Android和WebGL平台上,正如上面注解所讲,你无法通过文件系统直接访问。Android提供了AssetsManager这个API去访问,反映到Unity层面,就是WWW(过时)或者UnityWebRequest。
一般来说,在Editor下打出来需要在构建过程直接进入游戏包体的AssetsBundle都会放入这个目录下。但是这个目录在Android上是只读的,所以你无法将新下载的AssetsBundle放入这个目录。如果要更新的AssetsBundle,还需要另一个目录——
PersistentDataPath 可以通过Application.persistentDataPath
获取路径。
/User/AppData/Local/Packages/<productname>/LocalState
/sdcard/Android/data/<packagename>/files
在Android Studio中新建一个项目,然后在任何目录上右键 -> New -> Folder -> AssetsFolder。这会帮你创建一个assets目录。注意,你无论在哪个目录右键,新生成的assets目录都创建到/src/main
下。同理,选择New一个Raw Resources Folder,目录结构应该如下:
注意assets和res文件右下角都有一个小图标,表示这两个目录是属于资源目录。而著名的Android清单文件AndroidManifest.xml就位于res目录下。这个目录下的资源都会被编译,想要查看apk里面AndroidMenifest.xml的内容,可以通过Android Studio -> File -> Profile or debug APK 查看。
下面我们要创建几个Unity工程,查看他们的apk的目录结构的差别。
我在Resources下放置了一个内置的Standard Surface Shader,原大小1.7kB。到了apk里面就变成了36.7kB,说明Unity对其进行了编译,生成了完整的目标平台图形学API代码(OpenGL ES)。
因为是编辑器手动构建,所以构建代码需要放到Editor目录下。随意建一个Editor目录,然后新建一个脚本放到Editor目录下,内容为:
using UnityEditor; using System.IO; using UnityEngine; public class CreateAssetsBundle { [MenuItem("Assets/Build AssetBundles")] static void BuildAllAssetBundles() { string dir = Application.streamingAssetsPath; // AB包将放到StreamingAssetsPath下 if (!Directory.Exists(dir)) { Directory.CreateDirectory(dir); } // 开始打包 BuildPipeline.BuildAssetBundles(dir, BuildAssetBundleOptions.None, EditorUserBuildSettings.activeBuildTarget); } }
在Unity里随便建点东西,比如资源里弄点预置体,材质,着色器和贴图,然后场景中随便建两个物体,之后给Assets目录下的新东西赋予AssetsBundle的名称(在右下角)。我对应的文件目录和物体对应的AssetBundle Name如下(预置体采用的是下图中的Red材质球,这很重要,因为我们后面需要看到一些因此而带来的变化):
物体 | AssetBundle Name |
---|---|
Red(材质) | ab_mat |
RedShader(着色器) | ab_mat |
Cube(预置体) | ab_prefab |
Sphere(预置体) | ab_prefab |
SampleScene(场景文件) | ab_scene |
file(贴图 png) | ab_tex |
folder(贴图 png) | ab_tex |
然后点击 Assets -> Build AssetBundles 开始打包。打完包之后就会出现上图中的StreamingAssets目录。打开它你会发现这些东西:
可以发现,里面有两种文件,每种文件分别有一个AssetBundle文件,一个manifest清单文件
忽略.meta文件,因为这是Unity给任何Assets目录下的文件生成的身份信息文件,跟AssetBundle无关
manifest清单文件是文本文件,我们打开来看看其中预置体对应的AB包的清单文件的内容:
其实就是AssetsBundle的版本,一直是0,直到。。。Unity 2019。在Unity 2019以前,你都可以在Library目录下找到一个叫metadata的目录,是的,2019后就不见了。。。这又是另一个话题
HashAppended和ClassType不重要(其实是我也不懂。。。
我们将场景文件也打成了AB包,而且场景中有我们预置体的实例,所以查看场景AB包的清单文件,你也会在Dependencies下发现东西。
清单文件只是描述AB包的基础信息,而真正的资源都包含在AB包里。为了查看里面的内容,我们需要使用两个特殊的工具来解压和文本化。在Unity的安装目录的Editor/Data/Tools目录下会找到这两个工具:
将这个目录加入环境变量,以便我们能在命令行终端的任何路径下访问他们。打开命令行终端,定位到AB包所在的位置。然后先用WebExtract.exe尝试解压预置体所在的AB包:
得到一个新的 ab_prefab_data 目录,cd到里面,然后用binary2text.exe对里面唯一的一个文件进行文本化:
得到一个txt文件,打开它:
这虽然是一个纯文本文件,但是还有有一定结构的。推荐使用Sublime text查看,因为它提供了按照缩进进行代码折叠的功能,我们将所有缩进的文本折叠起来:
AB包的结构就变得一目了然了:
以下面的一个元数据为例:
ID
为-8598688866515239023,这里的ID
,也叫做的PathID
,是该资源在当前AB包中的唯一标识。ClassID
为4,即Transform。它是Transform这个类的唯一标识。Transform.gameObject
访问当前Transform类组件所在的物体的GameObject组件,所以这里出现了一个m_GameObject
。PPtr<GameObject>
表示一个指向GameObject类的指针。m_FileID
, m_PathID
用来表示当前这个属性其实是另一个资源,m_FileID
为0表示该资源在当前AB包中,否则需要在Header引入的AB包列表中对应查找。m_PathID
就是该资源的唯一标识符。x
,y
, z
, w
就是属性m_LocalRotation
的具体的值了,这个属性是一个四元数类。上面就是AB包的基本内容。然后,还要提及AB的类型,分成由场景文件Scene打包而来的场景AB包,以及普通资源打成的松散AB包。在松散AB包中,每一个包中都包含一个固定ID为1的,名字叫AssetBundle的资源。除了这个以外,其余所有的资源的ID都是一个绝对值很大的看起来很像是Hash的ID,这个ID一般来说是全局唯一的,如果两个AB包中包含同一个PathID的资源,就表示资源冗余了。而在场景AB包中,ID从1开始依次给场景中的资源计数,所以不同场景包中的资源的ID有重复自然就不奇怪了!另外,场景AB包在用WebExtract解压出来之后,是得到两个二进制文件,每个都需要单独用binary2text进行文本化。相比不同的AB包,场景AB包多出来的那个二进制文件是SharedAsset,描述当前场景中所有物体所共享的资源。一般来说,主要是材质,着色器以及与光照有关的资源。
binary2text.exe这个工具还可以用来文本化很多Unity的二进制文件,比如第一版元数据管理系统中,Unity会在Library/metadata下保存一些二进制文件,这些二进制文件其实就是当前工程中所有资源的序列化。通过binary2text.exe文本化这些二进制文件,你会看到很多类似的东西。
另外,binary2text还可以带一些参数,例如通过-detailed
参数可以使得文本化之后的文件附带很多额外的数据,包括表征当前资源大小的数据。
假设两个预置体被分别打进了两个AssetBundle。他们依赖同一个材质,材质使用一个贴图,然而材质和贴图都没有打AssetBundle。我们看看打出来的文件大小:
然后第二种方案是将材质也指定一个AssetBundle,再看看文件大小:
现象:材质没有打AssetBundle之前,两个预置体的AssetBundle大小都达到了87KB。材质打了AssetBundle之后,多了一个大小87KB的AssetBundle,但是两个预置体的大小都降低到了2KB。
为了内存的细粒度管理,项目喜欢把单个预置体打成一个AssetBundle。在这种情况下,上面第一种方案的内存会按照O(n)的复杂度增长,而第二种方案的内存增长的复杂度则是O(1)。
实际上,如果一个AssetBundle内的资源所依赖的另一个资源没有打AssetBundle,Unity会将其拷贝进AssetBundle里面。如果多个不同的AssetBundle的资源都依赖同一个没有打AssetBundle的资源,那么打包后该资源会在多个AssetBundle各存在一份拷贝。相反,如果将这个资源打包,那么那些依赖该资源的AssetBundle只会保存该资源所在的AssetBundle的引用。
Unity提供了一个插件叫AssetBundle Browser去查看工程里面的关于AssetBundle的信息。
下载地址:AssetBundles-Browser
将下载下来的文件整个目录拷贝到工程内的任何一个目录下,然后通过 Window -> AssetBundle Browser 打开。
AssetBundle Browser有三个页签,分别讲解:
Configure 页签有四个网格。
我当前选中的是一个预置体所在AssetBundle Name,可以看到有三个网格都打出Warning(警告)图标。提示的信息说明,当前AssetBundle Name所包含的资源有一些对外依赖,而这些依赖不从属于任何AssetBundle。右下角提示贴图Blender_icon被自动包含进了本AssetBundle中,而原本这个贴图是不从属于任何AssetBundle的。
打包构建页签。AssetBundle Browser已经帮你写好打包的代码了,使用方法略(因为一般项目都会制定自己的打包策略)。
这个页签会展示已经打成AssetBundle的包的内部信息。选择左上角的 Add Folder,选择AssetBundle所在的目录(比如我的是Assets/StreamingAssets)。然后在展示的文件列表里面选择一个AssetBundle,右边会展示AssetBundle内部的信息,包括它包含的资源,预加载的组件和依赖,等等。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。