에셋 번들을 작업할 때 크게 두 가지 단계가 있습니다. 첫 번째는 서버나 디스크 위치에서 에셋 번들을 다운로드 하는 것으로, WWW 클래스에서 실시합니다. 두 번째는 응용 프로그램에서 사용할 에셋 번들에서 에셋을 로드하는 단계입니다. 다음은 C# 스크립트의 예제입니다.
using UnityEngine;
using System.Collections;
public class BundleLoader : MonoBehaviour{
public string url;
public int version;
public IEnumerator LoadBundle(){
while (!Caching.ready)
yield return null;
using(WWW www = WWW.LoadFromCacheOrDownload(url, version)) {
yield return www;
AssetBundle assetBundle = www.assetBundle;
GameObject gameObject = assetBundle.mainAsset as GameObject;
Instantiate(gameObject);
assetBundle.Unload(false);
}
}
void Start(){
StartCoroutine(LoadBundle());
}
}
이 스크립트는 게임 오브젝트에 컴포넌트로 추가된 것입니다. 에셋 번들은 다음과 같은 방법으로 로드됩니다.
간단한 설명을 위해 이번 예에서는 안정성 검사를 전혀 하지 않았습니다. 보다 온전한 예는 여기에서 확인합니다.
애플리케이션을 만드는 것은 반복적인 프로세스이기 때문에 에셋을 여러 번 수정할 가능성이 높습니다. 또 변경사항이 있을 때마다 에셋 번들을 테스트할 수 있도록 다시 빌드해야 합니다. 에디터에서 에셋 번들을 로드할 수는 있지만 권장하는 방식은 아닙니다. 대신 에디터에서 테스트할 때 에셋 번들의 사용과 재빌드를 피하기 위해 헬퍼 함수 Resources.LoadAssetAtPath를 사용해야 합니다. 이 함수는 에셋이 에셋 번들에서 로드한 것처럼 해주지만 빌드 프로세스를 건너뛰고 에셋은 항상 최신의 상태로 유지됩니다.
다음은 헬퍼 스크립트의 예제로, 에디터에서 실행하느냐 아니냐에 따라 에셋을 로드하는 데 사용할 수 있습니다. 이 코드를 AssetBundleLoader.cs라는 C# 스크립트에 넣습니다.
using UnityEngine;
using System.Collections;
public class AssetBundleLoader {
public Object Obj; // The object retrieved from the AssetBundle
public IEnumerator LoadBundle<T> (string url, int version, string assetName, string assetPath) where T : Object {
Obj = null;
#if UNITY_EDITOR
Obj = Resources.LoadAssetAtPath(assetPath, typeof(T));
if (Obj == null)
Debug.LogError ("Asset not found at path: " + assetPath);
yield break;
#else
WWW download;
if ( Caching.enabled ) {
while (!Caching.ready)
yield return null;
download = WWW.LoadFromCacheOrDownload( url, version );
}
else {
download = new WWW (url);
}
yield return download;
if ( download.error != null ) {
Debug.LogError( download.error );
download.Dispose();
yield break;
}
AssetBundle assetBundle = download.assetBundle;
download.Dispose();
download = null;
if (assetName == "" || assetName == null)
Obj = assetBundle.mainAsset;
else
Obj = assetBundle.Load(assetName, typeof(T));
assetBundle.Unload(false);
#endif
}
}
이제 이 AssetBundleLoader 스크립트를 사용하면, 빌드한 애플리케이션을 실행하는 경우 에셋 번들에서 에셋을 로드하고, 에디터를 실행할 때는 프로젝트 폴더에서 에셋을 직접 로드할 수 있습니다.
using UnityEngine;
using System.Collections;
public class ExampleLoadingBundle : MonoBehaviour {
public string url = "http://www.mygame.com/myBundle.unity3d"; // URL where the AssetBundle is
public int version = 1; // The version of the AssetBundle
public string assetName = "MyPrefab"; // Name of the Asset to be loaded from the AssetBundle
public string assetPath; // Path to the Asset in the Project folder
private Object ObjInstance; // Instance of the object
void Start(){
StartCoroutine(Download());
}
IEnumerator Download () {
AssetBundleLoader assetBundleLoader = new AssetBundleLoader ();
yield return StartCoroutine(assetBundleLoader.LoadBundle <GameObject> (url, version, assetName, assetPath));
if (assetBundleLoader.Obj != null)
ObjInstance = Instantiate (assetBundleLoader.Obj);
}
void OnGUI(){
GUILayout.Label (ObjInstance ? ObjInstance.name + " instantiated" : "");
}
}
앞의 스크립트를 Assets 폴더의 ExampleLoadingBundle.cs 파일에 저장해야 합니다. public 변수를 올바른 값으로 설정하고 실행하면 에셋을 로드하는 데 AssetBundleLoader 클래스를 사용합니다. 그 후에는 인스턴스화 되고 GUI를 사용해 표시합니다.
You can use WWW.LoadFromCacheOrDownload which automatically takes care of saving your AssetBundles to disk. Be aware that on the Webplayer you are limited to 50MB in total (shared between all webplayers). You can buy a separate caching license for your game if you require more space.
에셋 번들이 StreamingAssets 폴더에 압축되지 않은 에셋 번들로 저장되어 있으면 디스크의 에셋 번들을 참조하는 데 AssetBundle.CreateFromFile을 사용할 수 있습니다. StreamingAssets 폴더의 에셋 번들이 압축되어 있으면 캐시에 있는 에셋 번들의 압축되지 않은 복사본을 생성하기 위해 WWW.LoadFromCacheOrDownload를 사용해야 합니다.
일부 플랫폼에서 에셋 번들이 호환됩니다. 다음 표를 가이드라인으로 활용하세요.
플랫폼 간 에셋 번들 호환성 | |||||
스탠드얼론 | Webplayer | iOS | Android | ||
에디터 | Y | Y | Y* | Y* | |
스탠드얼론 | Y | Y | |||
Webplayer | Y | Y | |||
iOS | Y | ||||
Android | Y |
For example, a bundle created while the Webplayer build target was active would be compatible with the editor and with standalone builds. However, it would not be compatible with apps built for the iOS or Android platforms.
(*)모바일 플랫폼용으로 빌드된 에셋 번들은 에디터가 실행되고 있는 플랫폼과 호환되지 않는 특정 플랫폼에 최적화된 포맷으로 저장된 데이터를 포함할 수도 있습니다. 발매된 게임에는 플랫폼마다 서로 다른 에셋 번들이 필요하다고 가정하는 편이 안전합니다. 특히 셰이더는 플랫폼마다 다릅니다.
E.g. GI data is stored in a format optimized for ARM architecture and thus is not loadable on x86 CPUs. Shaders are stored for OpenGLES 2 configuration and are not loadable by the Editor which uses the DX9/11 renderer (but works with OpenGLCore renderer). During development, you are recommended to use Simulation Mode mentioned in AssetBundles and the AssetBundle Manager tutorial to avoid Editor and mobile platforms incompatibility.
에셋 번들을 빌드할 때 내부적으로는 확장자를 제외한 파일 이름으로 에셋을 식별합니다. 예를 들어, Project 폴더의 “Assets/Textures/myTexture.jpg”에 위치한 텍스처는 기본 메서드를 사용할 경우 “myTexture”로 확인 및 로드됩니다. 에셋 번들을 BuildPipeline.BuildAssetBundleExplicitAssetNames를 이용해 빌드하면 오브젝트별로 자체적인 ID(문자열) 배열을 제공하여 이보다 더 많은 제어를 할 수 있습니다.
에셋 번들은 서로 다른 게임 간에 콘텐츠를 공유하게 해줍니다. 그러려면 에셋 번들의 게임 오브젝트로 참조되는 모든 에셋이 에셋 번들에 포함돼야 합니다. 참조된 에셋이 빌드되었을 때 에셋 번들에 확실하게 포함되게 하려면 BuildAssetBundleOptions.CollectDependencies 옵션을 전달하면 됩니다.
AssetBundles can contain a structure called a type tree which allows information about asset types to be understood correctly between different versions of Unity. On desktop platforms, the type tree is included by default but can be disabled by passing the BuildAssetBundleOptions.DisableWriteTypeTree to the BuildAssetBundle function. Webplayers intrinsically rely on the type tree and so it is always included (ie, the DisableWriteTypeTree option has no effect). Type trees are never included for mobile and console asset bundles and so you will need to rebuild these bundles whenever the serialization format changes. This can happen in new versions of Unity. (Except for bugfix releases) It also happens if you add or remove serialized fields in monobehaviour’s that are included in the asset bundle. When loading an AssetBundle Unity will give you an error message if the AssetBundle must be rebuilt.
에셋 번들에서 모든 오브젝트를 포함하는 배열을 조회하는 데 AssetBundle.LoadAllAssets를 활용할 수 있습니다. 그러나 식별자 리스트를 직접 얻을 수는 없습니다. 일반적인 해결책은 에셋 번들에 있는 에셋들의 이름을 갖고 있는 별도의 TextAsset을 두는 것입니다.
번들 A가 머티리얼과 텍스처를 가진 프리팹을 포함한다고 가정해보겠습니다. 번들 B에는 동일한 프리팹의 인스턴스를 가진 씬이 있습니다. 번들B는 해당 머티리얼에 액세스해야 합니다. 이 머티리얼이 특정 에셋 번들에 포함되지 않으면 A와 B 번들 모두에 포함됩니다.