Unity3D研究院之Assetbundle的实战(六十三)

1.创建Assetbundle

         无论是模型资源还是UI资源,最好是先把他们放在Prefab中,然后在做成Assetbundle。我们以模型来举例,Assetbundle中可以放一个模型、也可以放多个模型,它是非常灵活了那么最需要考虑的就是模型空间占用的问题。

比如我们有两个完全一样的模型,但是他们身上绑定的脚本不一样,此时需要把这两个模型放在两个不同Prefab中。如下图所示,我们分别对这两个Prefab打包,我们可以清晰的看到两个相同的Prefab打包在一起只占1M空间,而将他们分别打包会占1 + 1 = 2M空间。 Prefab在打包的同时会把模型身上的所有材质、贴图、组件、脚本全部包含进去。

         由此可得相同的模型尽量打包在一起,他们会公用一套资源文件。不相同的模型尽量分开打包,相同模型具有不同的脚本、组件的话把他们放在不同的Prefab中,最后把这些Prefab一起打包在一个Assetbundle中。如下图所示,现在Project视图中选择需要打包的Prefab,然后在导航菜单栏中选择Create Assetbundles Main表示分别打包、Create AssetBundles All表示将他们打包在一起。


        这两个prefab文件都指向了同一个模型,为了让它俩有所区别,我给它俩绑定了相同的脚本,但是脚本中的参数是不同的。在编辑器上给每个Prefab赋值一个不同的名子,然后在Awake方法中进行输出。

1

2

3

4

5

6

7

8

9

10

11

12

13

using UnityEngine;

using System.Collections;

public class Script : MonoBehaviour

{

public string name;

void Awake ()

{

Debug.Log("my name is "+ name);

}

}

Create Assetbundles Main : 分开打包,会生成两个Assetbundle。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

[MenuItem("Custom Editor/Create AssetBunldes Main")]

static void CreateAssetBunldesMain ()

{

//获取在Project视图中选择的所有游戏对象

Object[] SelectedAsset = Selection.GetFiltered (typeof(Object), SelectionMode.DeepAssets);

//遍历所有的游戏对象

foreach (Object obj in SelectedAsset)

{

string sourcePath = AssetDatabase.GetAssetPath (obj);

//本地测试:建议最后将Assetbundle放在StreamingAssets文件夹下,如果没有就创建一个,因为移动平台下只能读取这个路径

//StreamingAssets是只读路径,不能写入

//服务器下载:就不需要放在这里,服务器上客户端用www类进行下载。

string targetPath = Application.dataPath + "/StreamingAssets/" + obj.name + ".assetbundle";

if (BuildPipeline.BuildAssetBundle (obj, null, targetPath, BuildAssetBundleOptions.CollectDependencies)) {

Debug.Log(obj.name +"资源打包成功");

}

else

{

Debug.Log(obj.name +"资源打包失败");

}

}

//刷新编辑器

AssetDatabase.Refresh ();

}

最核心的方法其实就它:

BuildPipeline.BuildAssetBundle (obj, null, targetPath, BuildAssetBundleOptions.CollectDependencies)

参数1:它只能放一个对象,因为我们这里是分别打包,所以通过循环将每个对象分别放在了这里。

参数2:可以放入一个数组对象。

默认情况下打的包只能在电脑上用,如果要在手机上用就要添加一个参数。

Android上:

BuildPipeline.BuildAssetBundle (obj, null, targetPath, BuildAssetBundleOptions.CollectDependencies,BuildTarget.Android)

IOS上:

BuildPipeline.BuildAssetBundle (obj, null, targetPath, BuildAssetBundleOptions.CollectDependencies,BuildTarget.iPhone)

另外,电脑上和手机上打出来的Assetbundle不能混用,不同平台只能用自己的。

Create AssetBundles All:将所有对象打包在一个Assetbundle中。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

[MenuItem("Custom Editor/Create AssetBunldes ALL")]

static void CreateAssetBunldesALL ()

{

Caching.CleanCache ();

string Path = Application.dataPath + "/StreamingAssets/ALL.assetbundle";

Object[] SelectedAsset = Selection.GetFiltered (typeof(Object), SelectionMode.DeepAssets);

foreach (Object obj in SelectedAsset)

{

Debug.Log ("Create AssetBunldes name :" + obj);

}

//这里注意第二个参数就行

if (BuildPipeline.BuildAssetBundle (null, SelectedAsset, Path, BuildAssetBundleOptions.CollectDependencies)) {

AssetDatabase.Refresh ();

} else {

}

}

 两次打包完毕后,在StreamingAssets文件夹中就看到了这三个assetbundle文件。

2.读取Assetbundle 

       然后我们来学习如何运行时读取Assetbundle,Assetbundle是可以同时放在服务器或者本地的,无论放在哪里两种下载读取的方式是完全一样的。所以我建议在做unity项目的时候开始就把资源放在Assetbundle中在本地来做,等做的差不多了直接把Assetbundle放在服务器上,因为两种读取的方式完全一样,这样以后更新资源会方便很多。然后是读取,并且加载到游戏中。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

using UnityEngine;

using System.Collections;

public class RunScript : MonoBehaviour

{

//不同平台下StreamingAssets的路径是不同的,这里需要注意一下。

public static readonly string PathURL =

#if UNITY_ANDROID

"jar:file://" + Application.dataPath + "!/assets/";

#elif UNITY_IPHONE

Application.dataPath + "/Raw/";

#elif UNITY_STANDALONE_WIN || UNITY_EDITOR

"file://" + Application.dataPath + "/StreamingAssets/";

#else

string.Empty;

#endif

void OnGUI()

{

if(GUILayout.Button("Main Assetbundle"))

{

StartCoroutine(LoadMainGameObject(PathURL + "Prefab0.assetbundle"));

StartCoroutine(LoadMainGameObject(PathURL +  "Prefab1.assetbundle"));

}

if(GUILayout.Button("ALL Assetbundle"))

{

StartCoroutine(LoadALLGameObject(PathURL + "ALL.assetbundle"));

}

}

//读取一个资源

private IEnumerator LoadMainGameObject(string path)

{

WWW bundle = new WWW(path);

yield return bundle;

//加载到游戏中

yield return Instantiate(bundle.assetBundle.mainAsset);

bundle.assetBundle.Unload(false);

}

//读取全部资源

private IEnumerator LoadALLGameObject(string path)

{

WWW bundle = new WWW(path);

yield return bundle;

//通过Prefab的名称把他们都读取出来

Object  obj0 =  bundle.assetBundle.Load("Prefab0");

Object  obj1 =  bundle.assetBundle.Load("Prefab1");

//加载到游戏中

yield return Instantiate(obj0);

yield return Instantiate(obj1);

bundle.assetBundle.Unload(false);

}

}

 这里我们详细的说说 下载类WWW

WWW bundle = new WWW(path);

这样的做法是通过一个路径进行下载(无论是服务器路径还是本地路径下载操作都一样)但是bundle只能保存在内存中,也就是退出游戏在进入还得重新下,很显然在游戏中我们不能使用这种方式。

1

2

3

4

5

6

7

8

9

10

11

private IEnumerator LoadMainCacheGameObject(string path)

{

WWW bundle = WWW.LoadFromCacheOrDownload(path,5);

yield return bundle;

//加载到游戏中

yield return Instantiate(bundle.assetBundle.mainAsset);

bundle.assetBundle.Unload(false);

}

使用的方法是WWW.LoadFromCacheOrDownload(path,5);

参数1:服务器或者本地下载地址

参数2:版本号

         Unity会下载Assetbundle本地中,它的工作原理是先通过(版本号和下载地址)先在本地去找看有没有这个Assetbundle,如果有直接返回对象,如果没有的话,在根据这个下载地址重新从服务器或者本地下载。这里版本号起到了很重要的作用,举个例子,同一下载地址版本号为1的时候已经下载到本地,此时将版本号的参数改成2 那么它又会重新下载,如果还保持版本号为1那么它会从本地读取,因为本地已经有版本号为1的这个Assetbundle了。你不用担心你的资源本地下载过多,也不用自己手动删除他们,这一切的一切Unity会帮我们自动完成,它会自动删除掉下载后最不常用的Assetbundle ,如果下次需要使用的话只要提供下载地址和版本后它会重新下载。

        我们在聊聊Assetbundle 中的脚本,在移动平台下Assetbundle里面放的脚本是不会被执行的,还记得我们打包前给两个Prefab挂上了脚本吗?在手机上将Assetbundle下载到本地后,加载进游戏中Prefab会自动在本地找它身上挂着的脚本,他是根据脚本的名来寻找,如果本地有这条脚本的话,Prefab会把这个脚本重新绑定在自身,并且会把打包前的参数传递进来。如果本地没有,身上挂的条脚本永远都不会被执行。

      在Prefab打包前,我在编辑器上给脚本中的变量 name 赋了不同值,当Prefab重新载入游戏的时候,它身上脚本的参数也会重新输出。

如果你的Assetbundle中的Prefab上引用的对象,那么这样做就会出错了,你需要设定他们的依赖关系。或者运行时通过脚本动态的载入对象。

http://docs.unity3d.com/Documentation/ScriptReference/BuildPipeline.PopAssetDependencies.html

http://docs.unity3d.com/Documentation/ScriptReference/BuildPipeline.PushAssetDependencies.html

像这样重新打包就可以。

3.打包场景

     上面我们说过了打包Prefab,其实我们还可以把整个场景进行打包,因为移动平台不能更新脚本,所以这个功能就会有所限制,我的建议是烘培场景、然后把多个场景可复用的对象移除,场景中只保留独一无二的游戏对象,然后在打包场景,运行游戏时载入场景后,在动态的将之前移除的对象重新添加进来。

可以参考 : Unity3D研究院之将场景导出XML或JSON或二进制并且解析还原场景(四十二)

1

2

3

4

5

6

7

8

9

10

11

[MenuItem("Custom Editor/Create Scene")]

static void CreateSceneALL ()

{

//清空一下缓存

Caching.CleanCache();

string Path = Application.dataPath + "/MyScene.unity3d";

string  []levels = {"Assets/Level.unity"};

//打包场景

BuildPipeline.BuildPlayer( levels, Path,BuildTarget.WebPlayer, BuildOptions.BuildAdditionalStreamedScenes);

AssetDatabase.Refresh ();

}

          不同平台下需要选择  BuildTarget.Android 和 BuildTarget.iPhone 。 切记这段代码是把Level.unity常见文件打包到MyScene.unity3d文件中,所以在解包的时候也应当是先解开MyScene.unity3d,然后在去加载Level.unity场景,无需在ProjectSetting中注册新场景。

1

2

3

4

5

6

7

private IEnumerator LoadScene()

{

WWW download = WWW.LoadFromCacheOrDownload ("file://"+Application.dataPath + "/MyScene.unity3d", 1);

yield return download;

var bundle = download.assetBundle;

Application.LoadLevel ("Level");

}

          在测试情况下你可能会频繁的打包生成Assetbundle,如果忘记改版本号的话可能会读取之前的缓存,可能就会看不到新的效果,所以我建议在bunild Assetbundle的时候强制清空一下缓存。

Caching.CleanCache();

最后点击按钮进行加载Assetbundle和 Scene吧。

最后是下载地址:http://vdisk.weibo.com/s/Hrvea

欢迎大家一起学习,欢迎给我留言、欢迎一起讨论,祝大家学习愉快 啦啦啦啦。

2014年10月补充

WWW.LoadFromCacheOrDownload 这个方法建议大家以后不要再用了

因为是异步方法,而且还占用内存。

强烈建议使用AssetBundle.CreatFromFile 它是一个同步方法。现在IOS 和 android 都支持了,强烈建议用。

打包的时候需要选择不压缩。

1

2

3

4

//打包场景

BuildPipeline.BuildStreamedSceneAssetBundle(levels, path, target, BuildOptions.UncompressedAssetBundle))

//打包资源

BuildPipeline.BuildAssetBundle(null, assets, path, BuildAssetBundleOptions.UncompressedAssetBundle | BuildAssetBundleOptions.CollectDependencies, target);

因为不压缩, 所以就需要我们自己来压缩资源了, 可以用LZMA 和 GZIP来进行压缩。

1.打包出来的Assetbundle我们自己用LZMA压缩,上传到服务器上。

2.IOS或者Android下载这些assetbundle

3.解压缩这些assetbundle并且保存在Application.persistentDataPath 目录下。

4.以后通过AssetBundle.CreatFromFile  读取assetbundle。

此法确实可行,我们已经在实际项目中轰轰烈烈的使用了。。

时间: 2024-08-08 09:37:57

Unity3D研究院之Assetbundle的实战(六十三)的相关文章

Unity3D脚本中文系列教程(十六)

Unity3D脚本中文系列教程(十五) ◆ function OnPostprocessAudio (clip:AudioClip):void 描述:◆  function OnPostprocessGameObjectWithUserProperties (root : GameObject, propNames : string[], values : object[]) : void 描述:在导入文件时,为每个至少附加了一个用户属性的游戏物体调用propNames是一个string[ ],

Unity3D研究院之Jenkins的使用(七十八)

长夜漫漫无心睡眠,来一篇嘿嘿.我相信如果已经用Shell脚本完成IOS和Android打包的朋友一定需要Jenkins 怎么才能让策划打包ipa和apk?怎么才能彻底省去程序的时间,只要在同一局域网内不需要unity的开发环境,只要它有浏览器,它就能打包Jenkins无疑是最佳选择. Unity3D研究院之IOS全自动编辑framework.plist.oc代码(六十七) Unity3D研究院之IOS全自动打包生成ipa(六十八) Unity3D研究院之Android全自动打包生成apk(六十九

Unity3D手机斗地主游戏开发实战(02)_叫地主功能实现(不定期更新中~~~)

系列目录 Unity3D手机斗地主游戏开发实战(01)_发牌功能实现 Unity3D手机斗地主游戏开发实战(02)_叫地主功能实现 一.大体思路 前面我们实现了点击开始游戏按钮,系统依次给玩家发牌的逻辑和动画,并展示当前的手牌.这期我们继续实现接下来的功能--叫地主. 1.首先这两天,学习了DOTween,这是一个强大的Unity动画插件,大家可以参考:官方文档,个人感觉DOTween还是比较好用的. 好的,我们先来重构一下动画部分的代码(没有绝对牛逼的架构和设计,项目过程中不要不断的持续改进嘛

QT开发(六十三)——QT事件机制分析

QT开发(六十三)--QT事件机制分析 一.事件机制 事件是由系统或者QT平台本身在不同的时刻发出的.当用户按下鼠标.敲下键盘,或者是窗口需要重新绘制的时候,都会发出一个相应的事件.一些事件在对用户操作做出响应时发出,如键盘事件等:另一些事件则是由系统自动发出,如计时器事件. 事件的出现,使得程序代码不会按照原始的线性顺序执行.线性顺序的程序设计风格不适合处理复杂的用户交互,如用户交互过程中,用户点击"打开文件"将开始执行打开文件的操作,用户点击"保存文件"将开始执

Unity3D游戏开发从零单排(六) - 人物运动及攻击连击

提要 今天要实现的是一个简单人物控制器.包括用w,a,s,d来控制人物上下左右跑动,鼠标左击发出连招,都是基于老的lagacy的动画.虽然unity3d自带有charactorcontroller,但是并不是很好用,所以人物控制相关的全部自己来实现.先上效果图: 场景搭建 首先下载这个package,里面包含了人物的动作还有地面的模型.将人物和地面都拖进场景中.这里的模型默认的动画模式是lagacy,不用修改.模型有点偏小,改变模型的scale值为10.最好不要改源文件的scale的scale

《道德经?第六十三章》体悟

释道<道德经•第六十三章> 今听碧游宫通天教主讲道:“是以圣人犹难之,故终无难矣.”深有所感,遂查其出处.原文出自<道德经>,又名<老子>. 原文 为无为,事无事,味无味.大小多少,图难于其易,为大于其细:天下难事,必作于易,天下大事,必作于细.是以圣人终不为大,故能成其大.夫轻诺必寡信,多易必多难.是以圣人犹难之,故终无难矣. 译文 以无为的态度去有所作为,以不滋事的方法去处理事物,以恬淡无味当作有味.大生于小,多起于少.处理问题要从容易的地方入手,实现远大要从细微的

[转]从第六十三卦到第六十四卦

<易经>共有六十四卦,第六十三卦是“水火即济”卦,“济”是完成之意,如“直挂云帆济沧海”中的“济”,而“即济”是已经完成之意.“即济”卦也是六十四卦中最完美的一卦,为什么这么说呢?在六十四卦中每一卦均有六横组成,成为六爻,从上往下依次为第一爻……第六爻.而古人认为奇数为阳偶数为阴,所以第一.三.五爻的位置为阳位,第二.四.六爻的位置为阴位.而“――”为阴爻,“─”为阳爻,在“即济”中阳位全为阳爻,阴位全为阴爻.所以说“即济”是完美的一卦,它代表一切事情都已完成.在人体中,水在上火在下,水火相济

Unity3D研究院之IOS本地消息通知LocalNotification的使用

原地址:http://www.xuanyusong.com/archives/2632   现在的游戏里一般都会有本地消息,比如每天定时12点或者下午6点告诉玩家进入游戏领取体力.这种东西没必要服务器去推送,客户端就可以完成.Unity里面提供了本地任务的功能但是只有IOS上才支持,开始我有点不解为什么Android上不支持,当我把Android的本地通知做完后,我才明白.IOS源生的API中就支持固定时间循环推送,而Android上需要自己开启一个Services,启动一个AlarmManag

Unity3D研究院之IOS全自动打包生成ipa

接着上一篇文章, 自动生成framework,这篇文章我把shell自动化打包ipa整理了一下,希望大家喜欢,嘿嘿.. 建议大家先看一下上一篇文章.http://www.xuanyusong.com/archives/2720 首先我们要先搞清楚nity全自动打包的重要步骤. 1.自动生成xcode工程. 2.自动生成.ipa和dsym文件. 3.上传appstore(本篇略) 首先我们在做渠道包时,一般每个渠道都有自己一些特殊的需求,比如 游戏名子 .游戏图标.SDK.等等.那么我在在做自动化