Unity

[Unity] ScriptableObject ( 스크립터블 오브젝트)

EveryDay.DevUp 2020. 5. 22. 23:14

● 스크립터블 오브젝트의 정의

- 유니티 오브젝트를 상속받은 클래스로 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;
}

[ Monster Stat을 작성하면 우클릭 시 Asset 생성 메뉴가 만들어짐 ]

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 );
	}
}

[ 각 프리팹에 해당 컴포넌트를 붙이고 MonsterStat ScriptableObject를 연결 시킴 ]
[ 3개의 오브젝트 모두 동일한 데이터 값을 참조할 수 있음을 확인 가능]

4. 만약 연결한 데이터의 값을 변경하는 경우, 변경된 데이터 값이 참조하는 모든 오브젝트에 동일하게 적용 됨

[ Move Speed를 10 -> 999 로 변경한 경우 ]
[ 3개의 오브젝트 모두 동일하게 변경됨을 확인 가능 ]

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);
	}
}

[ MonsterStat Scriptable Object 를 참조 ]
[ Move Speed와 view는 Monster Stat에 있는 값을 가져오게 됨 ]

ScriptableObject의 의견

▶ ScriptableObject는 기존에 없던 시스템을 Unity가 새로 만든 것은 아님. Json, xml, csv 등으로 관리하던 데이터를 Unity에서 쉽게 사용할 수 있도록 편의적인 기능으로 제공하는 것으로 이해할 수 있다.

▶ 소규모의 데이터를 편하게 관리할 수 있는 장점이 있지만, 대용량의 데이터를 ScriptableObject로 관리하기엔 어려움이 있을 것으로 생각된다. 대규모의 데이터는 기본적으로 엑셀로 관리하는 것이 편하기 때문이다. 만약 ScriptableObject로 데이터를 관리한다고 할 경우 프로젝트에 맞는 툴 개발이 필요할 것이다.

▶외부 데이터를 파싱해서 사용하는 번거로움을 피하고, 데이터의 관리 부하가 크지 않다면 ScriptableObject를 사용하는 것이 편하고 최적화에도 도움이 될 것으로 생각한다. 또는 에디터에서 편하게 관리가 필요한 데이터만 ScriptableObject를 사용하는 것도 좋은 개발 방향이 될 수 있을 것으로 생각된다.

참고 자료 : https://www.youtube.com/watch?v=VtuSKmfrFDU