[Unity] ScriptableObject ( 스크립터블 오브젝트)
● 스크립터블 오브젝트의 정의
- 유니티 오브젝트를 상속받은 클래스로 Unity 시리얼라이즈 시스템을 가지고 있음
: 시리얼라이즈는 클래스나 오브젝트를 스트링이나 일차원 배열 형태로 변환하는 것을 얘기함
: 실제 파일을 열어보면 저장된 텍스트 내용을 확인할 수 있음
%YAML 1.1
%TAG !u! tag:unity3d.com,2011:
--- !u!114 &11400000
MonoBehaviour:
m_ObjectHideFlags: 0
m_CorrespondingSourceObject: {fileID: 0}
m_PrefabInstance: {fileID: 0}
m_PrefabAsset: {fileID: 0}
m_GameObject: {fileID: 0}
m_Enabled: 1
m_EditorHideFlags: 0
m_Script: {fileID: 11500000, guid: c9f7761a9a0c6f840a7561b294321975, type: 3}
m_Name: OrcStat
m_EditorClassIdentifier:
moveSpeed: 10
view: 20
atk: 50
def: 60
- MonoBehaviour보다 경량화된 데이터 컨테이너로써의 역할을 함
● MonoBehaviour와의 차이
- 유니티 콜백 중 OnEnable OnDisable OnDestroy 만 받음
: OnEable => ScriptableObject가 instantiated/loaded 될 때, ScriptableObject.CreateInstance() 함수 호출 시, 에디터에서 스크립트가 컴파일 된 후에 호출
: OnDisable => ScriptableObject가 파괴되기 전에 호출, 에디터에서 스크립트가 컴파일 되기 전에 호출
: OnDestroy => ScriptableObject가 파괴 될때 ( Object.Destroy )
- 게임오브젝트에 Add 컴포넌트 할 수 없음
- 어셋은 인스터싱이 되지 않고 고유한 하나의 파일로 저장됨
: 어셋의 내용을 변경하면 모든 참조하는 곳에 동일하게 변경된 내용이 반영됨
● ScriptableObject의 활용
- 데이터 컨테이너로 데이터 공유에 사용할 수 있고, 이를 통해 메모리 사용량 줄일 수 있음
: 기존 시스템은 하나의 컴포넌트에 변수를 정의하면 프리팹을 인스터싱해서 만드는 만큼 메모리 할당이 발생
: ScriptableObject는 데이터를 참조 형태로 오브젝트에서 가져올 수 있게 함으로써 메모리 할당을 최소화하고 고유한 값을 가질 수 있게함
- 프리펩에서 변화가 없는 동일한 데이터를 사용하는 경우, ScriptableObject로 해당 데이터를 관리하면 편리성과 최적화 효과를 얻을 수 있음
● 스크립터블 오브젝트의 사용
▶ 상단의 몬스터 예제를 실제로 Unity에서 구현하는 방법
1. ScriptableObject 코드 작성
using UnityEngine;
// 에디터에서 쉽게 사용할 수 있도록 메뉴를 만듬
[ CreateAssetMenu( fileName = "MonsterStat", menuName = "Scriptable Object Asset/MonsterStat" )]
public class MonsterStat : ScriptableObject
{
public int moveSpeed = 10;
public int view = 20;
}
2. MonsterStat 이라는 ScriptableObject를 Menu를 통해 생성
3. 사용하려는 오브젝트와 컴포넌트를 만들고 MonsterStat ScriptableObject를 연결 시킴
// Orc.cs
public class Orc : MonoBehaviour
{
public MonsterStat stat;
private void OnEnable()
{
Debug.LogWarning( " Orc - MoveSpeed : " + stat.moveSpeed + " view : " + stat.view );
}
}
// Human.cs
public class Human : MonoBehaviour
{
public MonsterStat stat;
private void OnEnable()
{
Debug.LogWarning( " Human - MoveSpeed : " + stat.moveSpeed + " view : " + stat.view );
}
}
// Elf.cs
public class Elf : MonoBehaviour
{
public MonsterStat stat;
private void OnEnable()
{
Debug.LogWarning( " Elf - MoveSpeed : " + stat.moveSpeed + " view : " + stat.view );
}
}
4. 만약 연결한 데이터의 값을 변경하는 경우, 변경된 데이터 값이 참조하는 모든 오브젝트에 동일하게 적용 됨
5. ScriptableObject는 상속해서 사용도 가능
using UnityEngine;
[CreateAssetMenu( fileName = "MonsterStat", menuName = "Scriptable Object Asset/Orc Stat" )]
public class OrcMonsterStat : MonsterStat
{
public int atk;
public int def;
}
6. 다른 ScriptableObject를 변수로 사용할 수 있어 데이터의 조합도 가능
using UnityEngine;
using System.Collections;
public class Orc : MonoBehaviour
{
public OrcMonsterStat stat;
private void OnEnable()
{
Debug.LogWarning( " Orc - MoveSpeed : " + stat.stat.moveSpeed + " view : " + stat.stat.view + " atk : " + stat.atk);
}
}
● ScriptableObject의 의견
▶ ScriptableObject는 기존에 없던 시스템을 Unity가 새로 만든 것은 아님. Json, xml, csv 등으로 관리하던 데이터를 Unity에서 쉽게 사용할 수 있도록 편의적인 기능으로 제공하는 것으로 이해할 수 있다.
▶ 소규모의 데이터를 편하게 관리할 수 있는 장점이 있지만, 대용량의 데이터를 ScriptableObject로 관리하기엔 어려움이 있을 것으로 생각된다. 대규모의 데이터는 기본적으로 엑셀로 관리하는 것이 편하기 때문이다. 만약 ScriptableObject로 데이터를 관리한다고 할 경우 프로젝트에 맞는 툴 개발이 필요할 것이다.
▶외부 데이터를 파싱해서 사용하는 번거로움을 피하고, 데이터의 관리 부하가 크지 않다면 ScriptableObject를 사용하는 것이 편하고 최적화에도 도움이 될 것으로 생각한다. 또는 에디터에서 편하게 관리가 필요한 데이터만 ScriptableObject를 사용하는 것도 좋은 개발 방향이 될 수 있을 것으로 생각된다.
참고 자료 : https://www.youtube.com/watch?v=VtuSKmfrFDU