各位朋友,大家好,欢迎大家关注我的博客,我是秦元培,我的博客地址是http://www.qinyuanpei.com。今天我们来说说通过反编译Unity3D的AssetBundle来提取游戏资源,博主写这篇文章的目的并非是要教大家如何去破解一款基于Unity3D引擎开发的游戏,而是想通过今天这篇文章来告诉大家如何在开发Unity3D游戏的过程中保护自己的游戏资源。
对于AssetBundle,其实博主是在以前的文章中是有提到的。不知道大家还记不记得,博主曾经在写游戏开发和Lua的不解之缘这个系列文章的时候,提到并且使用过AssetBundle这种技术。具体来说呢,AssetBundle在Unity3D中是一种用于资源打包盒资源动态加载的解决方法,比如我们平时玩的单机游戏容量一般都比较大,这是因为制作人员在制作游戏的时候将所有的项目资源都整合到了一起。可是如果我们用AssetBundle来做这个游戏的话,我们就可以只在发布的游戏中提供支持游戏功能的核心部分,而将游戏当中的场景、模型等资源以AssetBundle的形式打包然后放到服务器上,这样当游戏客户端处于联网的时候就可以从服务器上下载这些资源,从而实现游戏资源的动态加载,由此可见AssetBundle可以帮助我们减少游戏的容量。如果是在需要安装包的场合下,那么游戏包容量的大小无疑会为游戏加些印象分。
比如最近《轩辕剑6外传穹之扉》这部单机游戏发布了,从各大游戏网站的评测到和一样我喜欢单机游戏的各位朋友们的亲身体验,大家一致的认为这部游戏整体表现还不错,应该考虑玩一玩。这样难免让博主有些心动,可是看到17个G的游戏容量时还是犹豫了下。DOMO小组从《轩辕剑6》就开始使用Unity3D引擎,在经历了第一部游戏的失败后,或许此次DOMO小组会将游戏优化的比较好吧。这里如果有喜欢单机游戏的朋友不妨去玩玩看,毕竟我们学习游戏开发的初衷就是做出好游戏,如果不热爱游戏又怎么能做出好游戏呢?好了,扯得有点远了,这里我们注意到一个重要的因素就是游戏容量,如果DOMO采用AeestBundle的话,游戏的容量肯定会减少很多。可是这样一来,它就不是单机游戏了嘛,对吧!
在Unity3D中AssetBundle是专业版中的一个功能,在免费版的Unity3D中是无法使用这个功能的,不知道在Unity5中这个功能是不是划分到了个人版中。好了,下面我们来看看如何使用AssetBundle。我们主要从使用AssetBundle打包和加载AssetBundle这两个方面来说:
使用AssetBundle打包主要通过BuildPipeline.BuildAssetBundle()这个方法来实现,该方法原型为:
bool BuildAssetBundle (Object mainAsset,Object[] assets,string pathName, BuildAssetBundleOptions
optionsBuildAssetBundleOptions.CollectDependencies | BuildAssetBundleOptions.CompleteAssets,
BuildTarget targetPlatform= BuildTarget.WebPlayer)
在这个方法中,第一个参数是一个Object类型,表示一个激活的物体;第二个参数是一个Object[]类型,表示所有选中的物体;第三个参数是一个string类型,表示要导出的资源包的路径,资源包的扩展名可以是assetbundle或者unity3d;第四个参数表示的是打包选项,默认是完全打包和依赖打包。这里重点解释下这两个概念,完全打包是指所有资源都参与打包,比如说一个模型带有贴图和动画,那么打包模型的时候贴图和动画都会被作为资源打包。而依赖打包是相对于Prefab来说的,比如说PrefabA中引用了PrefabB这个对象,那么打包的时候这两个对象都会被打包,并且它们之间的这种依赖关系会在打包后继续保持;第五个参数是平台的选择,因为Unity3D是一个跨平台的游戏引擎,而各个平台现在的情况又不尽相同,因此现在Unity3D采取的方案是各个平台只能使用自己平台对应的AssetBundle,这一点希望大家在使用的时候注意啊。好了,现在我们来看一个简单的例子:
/// <summary>
/// 输出AssetBundle
/// </summary>
/// <param name="type">平台类型</param>
static void ExportToAssetBundle(ExportType type,BuildTarget target)
{
//获取存储路径
string savePath=EditorUtility.SaveFilePanel("输出为AssetBundle","","New Resource","unity3d");
if(savePath==string.Empty) return;
//获取选中的对象
Object[] selection=Selection.GetFiltered(typeof(Object),SelectionMode.DeepAssets);
if(selection.Length==0) return;
//打包
if(type==ExportType.All){
BuildPipeline.BuildAssetBundle(null,selection,savePath,BuildAssetBundleOptions.CollectDependencies,target);
}else{
BuildPipeline.BuildAssetBundle(obj,null,savePath,BuildAssetBundleOptions.CollectDependencies,target);
}
}
这是一个简单的导出AssetBundle资源包的方法,它有两个参数,第一个参数表示是一个枚举类型,定义为ExportType,取Single时表示打包一个特定的激活物体,比如说一个模型、一个场景等等;取All时表示打包所有选中的物体,比如一个场景。第二个参数表示打包的平台,这个不用多说了。因为博主的免费版的Unity3D不支持AssetBundle,所以这里没法给大家演示了,具体效果请自行测试,有问题的话给博主留言就是了。
加载AssetBundle是一个从网络中下载资源的过程,因此需要使用Unity3D的WWW功能,这是一个简单的网络协议的封装,可以像浏览器一样访问某个URL地址或者是本地地址,访问WEB地址需要使用HTTP协议,访问本地地址需要使用File协议。我们来看一个具体的例子:
/// <summary>
/// 加载一个unity3d格式的文件
/// WEB地址——http://server.com/xxx.unity3d
/// 本地地址——file://.unity3d文件的绝对路径
/// </summary>
IEnumerator LoadUnity3DFile(string url)
{
WWW www=new WWW(url);
yield return www;
if(www.error!=null){
Debug.Log(www.error);
}else{
AssetBundle bundle=www.assetBundle;
Instantiate(bundle.mainAsset,Vector3.zero,Quaternion.identity);
}
}
在这里我们直接使用bundle.assetBundle获取了全部的资源,如果只需要获取资源中的一部分,则只需要通过bundle.Load()方法就可以了,这里需要传入资源的名称。当我们使用完资源后可以通过bundle.Unload()方法来卸载资源,达到释放内存的目的。
好了,下面我们以《新仙剑OL》这款游戏的AssetBundle的反编译来探索下在使用AssetBundle打包应该注意哪些问题。《新仙剑OL》这款游戏呢,是采用Unity3D引擎开发的一款横跨客户端游戏和网页游戏的网络游戏,游戏以《仙剑奇侠传》初代游戏剧情为主,玩家将第三人称视角再次跟随主人公展开一段荡气回肠的感人故事。这款游戏总体来说还不错吧,因为毕竟是网游,我们不能用单机游戏的视角去评价,具体的原因大家都是知道的。
好了,为什么我们要选择这款游戏呢?
* 第一,这款游戏的客户端只有30余M,体积小适合拿来研究(这就是AssetBundle的好处啊)* 第二,博主是一位仙剑玩家,一直希望有一天《仙剑奇侠传1》能够用3D技术重现,这个游戏满足了博主的好奇心
* 第三,网络上已经有朋友对这个游戏的打包进行了研究,这里感谢网友朋友提供部分.unity3d文件及相关文件。
我们选择的解包工具是一款叫做disunity的命令行工具,经过博主的尝试,这个工具真心强悍啊,可以解开.unity3d文件和.assets文件,可以拿到的数据形式有贴图、声音、模型等。具体的情况大家可以在稍后看到。
首先我们找到《新仙剑OL》的安装目录,然后我们就能发现一个叫做assetbundles的文件夹,这是怕大家不知道吗?这太明显了吧!我们打开文件夹会发现Charachers、NPC、Scene等等文件夹,继续往下找我们发现了好多的.unity3d文件,不过这些文件都是以.unity3d然后跟些随机字符串的形式存在的。根据网友朋友们的提示,这些文件就是.unity3d文件,不过游戏制作组为了干扰我们故意接了下随机字符在后面(呵呵,还有比这更弱的加密方式吗?)。博主看到这里的第一感觉就是想先用加载AssetBundle的方式来看看能不能将这些AssetBundle读取出来,因此果断改了文件扩展名,然后开始在Unity3D中读取,结果程序报错看来是我们想的简单了啊。没办法的办法,强行解包吧!在命令行中输入:
disunity extract C:\Users\Robin\Desktop\s049.unity3d
接下来程序会在桌面上生成一个上s049的文件夹,打开文件夹一看,尼玛,竟然直接拿到了模型的网格数据(.obj)和贴图数据(.dds)以及相关的Shader。这让我突然间有点不能接受啊,马上打开Blender将网格数据导入,结果童年的林月如就出现在了我们的面前:
因为博主不会在Blender中给模型贴图,所以我们到Unity3D中完成贴图,首先需要将模型导出为FBX格式。好了,将模型导入Unity3D后,将贴图赋给模型,童年的林月如就闪亮登场了,哈哈!
好了,再来一张,不过这张没有贴图,需要大家自己来辨别这是谁啊,哈哈!
通过disunity这个工具我们还能获取更多的资源,剩下的内容就由大家自己去探索吧。通过这部分的研究,我们可以总结出以下观点,希望大家在使用AsssetBundle这项技术时注意:
* 尽量在一个AssetBundle中打包多个资源,这样做的好处是别人没法通过加载AssetBundle拿到你做好的Prefab。
* 尽量将一个预制件分割成不同的部分分别存放,这样做的好处是即使别人拿到了你的预制件却是不完整的。
* 尽量利用动态脚本来加载场景而不是将整个场景打包,即使将整个场景打包,要把贴图和模型分开放置(因此如此,我虽然拿到了游戏的场景贴图,可是没有用啊)
* 尽量利用加密的方法来隐藏本地的AssetBundle或者使用不易察觉的存储位置作为AssetBundle的存储位置,不要用明文数据进行存储。
好了,今天的内容就是这样了,希望大家喜欢,AssetBundle打包是一个值得去深入研究的问题,今天博主提出的这些观点不过是对《新仙剑OL》这个游戏的打包提出de一些看法,如果大家有不同的看法,欢迎一起来交流!