게임 개발을 공부하면 오브젝트 풀에 대해 배우면서

오브젝트를 미리 생성하고 필요할때 꺼내서 사용하고 사용이 다되면 풀에 반납하여 비활성화 시켜주는 것을

오브젝트 풀링이라고 배워왔다.

 

하지만 유니티에서는 이 오브젝트 풀을 사용 할수 있게 패키지로 제공하고 있다.

 

using UnityEngine.Pool;

해당 패키지를 사용하기 위해서는 해당 코드를 작성하고 사용해야된다.

 

해당 Pool를 사용하려면 어떡해 해야되는가?

public IObjectPool<GameObject> objectPool;
public ObjectPool<GameObject> pool;

변수로 선언할때는 이 두방식을 이용하여 변수를 초기화 하여 사용하면된다.

 

초기화시 필요한 생성자 매개변수들

 pool = new ObjectPool<GameObject>(CreateObject, GetObject, ReleaseObject, DestroyObject, collectionChecks, minSize, maxSize);
 objectPool = new ObjectPool<GameObject>(CreateObject, GetObject, ReleaseObject, DestroyObject, collectionChecks, minSize, maxSize);

이런식으로 선언을 하면된다. 

 

namespace UnityEngine.Pool
{
    public interface IObjectPool<T> where T : class
    {
        int CountInactive { get; }

        T Get();

        PooledObject<T> Get(out T v);

        void Release(T element);

        void Clear();
    }
}

 

해당 코드가 IObjectPool의 내부 구조이다.

 

using System.Collections.Generic;
using System.ComponentModel;
using UnityEditor.Presets;
using UnityEngine;
using UnityEngine.Pool;


//최소 50개의 오브젝트 수 보장, 부족할 경우 누적 300개까지 추가 생성, 300개가 넘어갈 경우 가장 오래전에 생성된 오브젝트를 반환 후 재사용
public class Week2_OjbectPool_Q4 : Singleton<Week2_OjbectPool_Q4>
{

    [SerializeField] private GameObject prefab;
    public ObjectPool<GameObject> pool;

    public bool collectionChecks = true;
    private const int minSize = 50;
    private const int maxSize = 300;

    private GameObject container;
    void Awake()
    {
        container = new GameObject(prefab.name + "_Container");

        /*
        createFunc: 오브젝트 생성 함수 (Func)
        actionOnGet: 풀에서 오브젝트를 가져오는 함수 (Action)
        actionOnRelease: 오브젝트를 비활성화할 때 호출하는 함수 (Action)
        actionOnDestroy: 오브젝트 파괴 함수 (Action)
        collectionCheck: 중복 반환 체크 (bool)
        defaultCapacity: ObjectPool의 List<T>에 미리 자리 만드는것(오브젝트 생성x)(int)
        maxSize: 저장할 오브젝트의 최대 갯수 (int)
         */
        pool = new ObjectPool<GameObject>(CreateObject, GetObject, ReleaseObject, DestroyObject, collectionChecks, minSize, maxSize);
    }

    private GameObject CreateObject()
    {
        // [요구스펙 1] Create Object

        //Instantiate();
        GameObject obj = Instantiate(prefab, container.transform);
        return obj;
    }

    public void GetObject(GameObject obj)
    {
        // [요구스펙 2] Get Object
        obj.gameObject.SetActive(true);
    }

    public void ReleaseObject(GameObject obj)
    {
        // [요구스펙 3] Release Object
        obj.gameObject.SetActive(false);
    }

    public void DestroyObject(GameObject obj)
    {
        //오브젝트 파괴 함수
        Destroy(obj.gameObject);
    }

    private void Update()
    {
        if (Input.GetKeyDown(KeyCode.Space))
        {
            for (int i = 0; i < 10; i++)
            {
                pool.Get();
            }
        }
    }
}

해당 코드는 UnityEngine.Pool을 이용하여 오브젝트 풀을 구성한 예제 코드이다. 

사용시에는 따로 풀링할 오브젝트를 넣어 등록해서 사용해야된다. 

 

 

해당 Docs를 참고하여 글을 작성하였습니다. 

https://docs.unity3d.com/ScriptReference/Pool.ObjectPool_1.html

 

Unity - Scripting API: ObjectPool<T0>

Object Pooling is a way to optimize your projects and lower the burden that is placed on the CPU when having to rapidly create and destroy new objects. It is a good practice and design pattern to keep in mind to help relieve the processing power of the CPU

docs.unity3d.com

 

https://01149.tistory.com/115

 

Unity :: UnityEngine에서 제공하는 Pool 패키지(오브젝트 풀)

게임 개발을 공부하면 오브젝트 풀에 대해 배우면서오브젝트를 미리 생성하고 필요할때 꺼내서 사용하고 사용이 다되면 풀에 반납하여 비활성화 시켜주는 것을오브젝트 풀링이라고 배워왔다.

01149.tistory.com

 

오늘 유니티 2D 팀프로젝트 하고 튜터님들이 각 팀들마다 피드백 해주셨는데

내 설계에도 도움이 될것 같아서 작성해본다.

 

0.프리팹화 한거 Resources를 사용하여 동적 생성하여 Scene 채우기 할것


1.매니저에 다 모든 기능을 넣지마라


2.Find 메서들 지양해라


3.문자열을 사용하는 메서드들을 해쉬코드로 변경하여 사용할것


4.오브젝트 충돌 검사시 태그 검사가 아닌
오브젝트 비트연산자 + 레이어마스크 검사를 하거나
컴포넌트를 호출하여 검사하도록 하자


5.이름이니셜로 폴더링 하지말것, 만약 그 팀원 퇴사했다 치면
어떤 기능일 구현했는지 모르기 때문에 매우 큰 문제가 된다.결국 코드 작업시 남의 스크립트 영역에 침범하게 될일이 있으므로 기능 별로 폴더링하여 사용하여야된다.


6.게임매니저가 Player를 찾게 하지말고
Player에서 게임매니저를 접근하게 하는 로직을 생성하자
ex)Player 오브젝트에서 게임 저장
GameManager.Instance.GameSave() ....


7.매직 넘버 지양하자

코드에 하드코딩으로 숫자 집어넣는거 


8.파입 입출력을 이용하자 스테이지 구성을 지향하자


9.프로젝트 진행할때 플랫폼 설정하기-> 안드로이드,IOS,윈도우 등등...


10.프로젝트 세팅에서 fixedUpdate의 시간을 조정 가능하다. 


11.GetComponent<>는 최적화가 잘되어있지만 수시로 불러 사용하는건 지양해야된다.


12.Dirctinoray의 키값을 enum으로 해서 가독성을 올릴것 


13.코루틴은 효과만 관리하고 플래그(변수)는 메서드에서 관리하는것이 좋다.


14.코루틴이 실행중인데 중복 실행 방지 로직은 항상 생각할것

15.어디서든 UI매니저를 통해서 호출하게하면 좋다.

 

각팀들에게 해주셨던 피드백을 간단하게 정리한 내용이다.

설계할때 한번씩 보고 설계하도록 하자 

Q.제네릭을 왜 사용하는걸까?

A.간단히 말해서 클래스를 자료형에 상관없이 자유롭게 쓸 수 있게 하는것

즉, 클래스를 변수처럼 사용 하고 싶을때 주로 사용한다.

정수,문자열 데이터 를 담는공간을 변수라 하고

함수를 변수처럼 사용하면 delegate를 사용하고

class를 변수처럼 사용하기 위해 제네릭을 사용한다. 

이런 느낌으로 보면 된다.

 

Q.제네릭을 쓰면 뭐가 좋아?

A. 클래스에 자료형 제약을 걸고 사용 하여 처음 구조 설계시 시간이 걸리지만

유지 보수에 매우 효율적이기 때문이다.

 

Q.어떨때 써??

A. 자주 사용되는 클래스인데 자료형이 다양한 클래스인경우에 주로 사용된다.

첫번쨰 예시 ) 탄막슈팅게임에서 총알,아이템,이펙트 등등 다양하게 객체를 생성해야 되는 류의 게임에서

오브젝트 풀링을 제너릭 구조로 설계해서 사용하기

 

자료형을 담을 오브젝트 Pool을 선언해 준다. 그리고 제약조건으로 MonoBehaviour를 선언하여 오브젝트 및 컴포넌트로 사용 할수 있는 class만 T로 선언 할 수있다.

public class ObjectPool<T> where T : MonoBehaviour
{
    public Queue<T> PoolQueue = new Queue<T>();

    public bool InitPushObject(T poolObject)
    {
        if (poolObject == null) 
        {
            Debug.Log(poolObject.name + "이 null 입니다.");
            return false;
        }

        poolObject.gameObject.SetActive(false);
        PoolQueue.Enqueue(poolObject);

        return true;
    }

    public T PoolObject(Vector2 pos)
    {
        if (PoolQueue.Count <= 5)
        {
            Debug.Log(typeof(T).Name + "오브젝트 갯수 부족!");

            return null;
        }

        T obj = PoolQueue.Dequeue();
        obj.gameObject.transform.position = pos;
        obj.gameObject.SetActive(true);
        return obj;
    }

    public void PushObject(T pushObj)
    {
        pushObj.gameObject.SetActive(false);
        PoolQueue.Enqueue(pushObj);
        //Debug.Log("총알 큐 갯수"+PoolQueue.Count);
    }



}

 

T로 해당 자료형을 선언하고 Dictionary의 value 값으로 ObjectPool<T>를 선언하여 

SpawnManager도 제네릭으로 설계한다.

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public abstract class SpawnManager<T> : MonoBehaviour, iManager where T : MonoBehaviour,iPoolable<T>
{
    //부모 클래스를 저장
    [SerializeField] protected List<T> prefabesList;

    protected Dictionary<string, ObjectPool<T>> objectPools = new Dictionary<string, ObjectPool<T>>();
    public virtual void Initialize() { }
}

 

결국 최종적으로 사용하는 EnemySpanwManager에 SpawnManager<EnemyController>를 상속하여 제네릭으로 생성하였다. 이렇게되면 자료형만 바꿔 작성하면 Item,Effect 생상선이 올라가기 때문이다.

 

또한 이 구조에서 T는 부모 클래스를 넣고서 EnemyController의 자식 클래스로 형변환하여 확장성도 확보가 가능해진다.

public class EnemySpawnManager : SpawnManager<EnemyController>
{
    [SerializeField] private ItemSpawnManager itemSpawnManager;
    [SerializeField] private EffectManager effectManager;
    public override void Initialize()
    {
        Debug.Log(gameObject.name + "Initalize 완료!");
    }

    private void Awake()
    {
        EnemyController gameObj;

        foreach (EnemyController prefab in prefabesList)
        {
            GameObject poolContainer = new GameObject("Pool_Container_" + prefab.name);

            ObjectPool<EnemyController> objectPool = new ObjectPool<EnemyController>();

            for (int i = 0; i < prefab.EnemySO.PoolCount; i++)
            {
                gameObj = Instantiate(prefab, poolContainer.transform);
                gameObj.OnEventDieObject += GameManager.Instance.GetScore;
                gameObj.OnEventPushObject += PushObject;
                gameObj.OnEventDropItem += itemSpawnManager.RandomPoolObject;
                gameObj.OnEventDieEffectObject += effectManager.PoolObject;
                objectPool.InitPushObject(gameObj);
            }

            objectPools.Add(prefab.EnemySO.enemyName, objectPool);
        }

        Initialize();
    }

    public EnemyController PoolObject(string objName, Vector2 spawnPos)
    {
        return objectPools[objName].PoolObject(spawnPos);
    }

    public EnemyController PoolObject(EnemyController enemy, Vector2 spawnPos)
    {
        return objectPools[enemy.EnemySO.enemyName].PoolObject(spawnPos);
    }

    private void PushObject(EnemyController enemy)
    {
        //objectPools[testmonster.monsterName].PushObject(testmonster);
        objectPools[enemy.EnemySO.enemyName].PushObject(enemy);
    }

}

 

예시2) 싱글톤 구조를 제네릭으로 구성 

싱글톤 구조는 Manager 클래스에서 정말 많이 쓰이기에 제네릭으로 구조를 잡고 상속으로 사용하게되면

효율성이 매우증가한다.

 

이 외에도 많은 사용용도가 있지만 나는 이 목적으로 많이 사용을 했던것 같다. 

오브젝트 풀링 만들려고 구조 잡는중에 
ObjectPool을 제네릭으로 만들고 게임오브젝트 생성하듯이 하려고했으나
자꾸 null값을 반환하여 왜그런지 찾아보니 
제네릭 클래스는 컴포넌트화하지 못한다 , 즉, 오브젝트로 못만든다는 뜻이다.
그래서 Monobehaviour를 해지하고 C#문법을 이용하여 Class를 호출하여 사용하여아한다. 

 

public class ObjectPool<T> : MonoBehaviour where T : Component
{
    public Queue<T> PoolQueue;

    public void Start()
    {
        PoolQueue = new Queue<T>();
    }

    public void Initailize(int poolCount, T poolObject)
    {
        GameObject poolContainer = new GameObject("Pool_Container_" + poolObject.name);

        T gameObj;
        for (int i = 0; i < poolCount; i++)
        {
            gameObj = Instantiate(poolObject, poolContainer.transform);
            gameObj.gameObject.SetActive(false);

            PoolQueue.Enqueue(gameObj);
        }
    }
}

 /*====================================================================*/

public class EnemySpawnManager : SpawnManager<TestMonster>
{
  
  private void Start()
    {
        foreach (TestMonster prefab in prefabesList)
        {
            prefab.OnEventPushObject += PushObject;
			
            //트러블슈팅이 난구간 objectPool을 자꾸 null을 반환한다.
            ObjectPool<TestMonster> objectPool = new ObjectPool<TestMonster>();
            
            objectPool.Initailize(prefab.PoolCount, prefab);
            objectPools.Add(prefab.name, objectPool);
        }

        Initialize();
    }
}

 

해당 오브젝트의 objectPool을 받아오려 했으나 자꾸 null을 반환해서 문제가 발생하였다.

 

//트러블슈팅 대처 : Mononehaviour 를 상속해지하였음 
public class ObjectPool<T> where T : MonoBehaviour
{
    public Queue<T> PoolQueue = new Queue<T>();
    
    public Queue<T> PoolQueue = new Queue<T>();
	
    //Initialize 메서드를 변경하였음 (트러블슈팅과 문제되는부분 x)
    public bool InitPushObject(T poolObject)
    {
        if (poolObject == null) 
        {
            Debug.Log(poolObject.name + "이 null 입니다.");
            return false;
        }

        poolObject.gameObject.SetActive(false);
        PoolQueue.Enqueue(poolObject);

        return true;
    }
}

/*===========================================================*/

public class EnemySpawnManager : SpawnManager<TestMonster>
{
	private void Awake()
	{
    	TestMonster gameObj;

        foreach (TestMonster prefab in prefabesList)
        {
            GameObject poolContainer = new GameObject("Pool_Container_" + prefab.name);
			
            //트러블슈팅 대처 : 
            //Monobehaviour를 상속해지해서 new 어트리뷰트를 이용해여 클래스 객체 생성을 함
            ObjectPool<TestMonster> objectPool = new ObjectPool<TestMonster>();

            for (int i = 0; i < prefab.PoolCount; i++)
            {
                gameObj = Instantiate(prefab, poolContainer.transform);
                gameObj.OnEventPushObject += PushObject;
                objectPool.InitPushObject(gameObj);
            }

            objectPools.Add(prefab.monsterName, objectPool);
        }

    	Initialize();
	}
}

ObjectPool의 Monobehaviour를 상속해지하고 C# 문법의 new로 객체를 생성한뒤 호출하여 사용하였다.

=============================2024.10.21 수정=============================================

 

제네릭을 상속 받은 클래스들 -> EnemySpawnManager는 이미 오브젝트로 만들어 진다.

하지만 ObjectPool<T>를 컴포넌트 및 오브젝트화해서 사용하지 못한다.

즉, 제네릭 클래스 그 자체로는 사용하지 못하고 상속을 해서 사용해야 올바른 사용법이다.

 

개발 도중 외부에서 접근해서 데이터나 객체를 가져와야되는 경우가 있는데

이렇게 될 경우 해당 매니저는 이 기능이 필요없는데 이 기능을 참조해서 가지고 있어야되는경우가 종종 생긴다.

 

ex) EnemyManager는 적만 관리/생성 하는 클래스인데 아이템드랍,적 파괴시 이펙트 효과 등 이벤트를 적용시키려다 보니 ItemManager와 EffectManager가 참조 되있는 경우

 

이런 경우를 대처하기위해 첫 설계를 잘해야되는데 어떡해 해야 되는가에 대한 정리이다.

해당 캠프 진행중 튜터님과의 면담으로 답변 받은것을 정리한 내용이다.

 

1.해당 Scene에서 사용되는 Data의 중간다리 역할을 해주는 Class를 만들어서 사용한다.

이 방법은 Scene 이동시에도 싱글톤 패턴을 사용하면 매우 효과적이며 의존관계가 아닌 연관관계로 멤버변수로 필요한 class들을 가지는 방법이다. 

 

2.이 방법은 해당 기능을 따로 class로 만들고 데이터를 보관하고 있는 컨테이너 클래스로 만들어 필요할떄마다 해당 클래스를 호출하여 쓰는 방법이다. 예시로 SpawnManager로 객체들을 생성하는 기능을 하나의 Class로 만들고 오브젝트 관리하는 Class를 만들어 해당 Class에서는 오브젝트 풀링처럼 객체의 data(prefab)만 꺼내서 전달해주는 방식이다. 

벡터의 내적은 두 벡터사이의 각도를 알아낼때 주로 사용한다.

그렇다고 내적을 해서 나온값이 두 벡터사이의 각도값이 아니다. 

 

이미지 출처 : https://moondongjun.tistory.com/129

벡터의 내적은 Dot Product라고도 불리며 두 벡터의 곱으로 스칼라 값을 가져올수 있고

해당 스칼라값을 ACos(theta)에 사용하면 두벡터의 사이각을 가져올 수있다.

 

Vector2.Dot()를 활용해보기 위해 간단한 예제 코드이다. 

using UnityEngine;

public class VectorAngle : MonoBehaviour
{
    public Vector2 v1 = new Vector2(1, 0);
    public Vector2 v2 = new Vector2(0, 1);

    void Start()
    {
        
        float dotProduct = Vector2.Dot(v1, v2);

        // .magnitude : 해당 벡터의 스칼라값(크기)를 반환함
        float magnitude1 = v1.magnitude;
        float magnitude2 = v2.magnitude;

        // dot내적값 = A크기 * B크기 * Cos(세타)
        // Cos(세타) = dot 내적값 / A크기 * B크기 로 식변환
        float cosAngle = dotProduct / (magnitude1 * magnitude2);

        // 구한 각도값을 이용하여 Acos에 넣고 사용 
        float angleInRadians = Mathf.Acos(cosAngle);
        //라다안으로 반환하기에 Rad2Deg 사용
        float angleInDegrees = angleInRadians * Mathf.Rad2Deg;

        
    }
}

이것으로 해당 벡터의 각도 값을 알아 낼 수 있다. 

https://01149.tistory.com/107

 

Unity2D :: Lerp , Slerp에 대하여 / Slerp로 객체 회전시키기

Q.Slerp가 뭐야?A.Slerp는 구면선형보간(Spherically interpolate)을 의미한다.구면선형보간은 두 지점 사이의 위치를 파악하는 선형보간(lerp)와 같지만 직선이 아닌 곡선으로 파악을한다. Q.Slerp가 무슨뜻

01149.tistory.com

https://01149.tistory.com/108

 

Unity :: Generic(제네릭) 클래스는 컴포넌트로 사용하지 못한다.

오브젝트 풀링 만들려고 구조 잡는중에  ObjectPool을 제네릭으로 만들고 게임오브젝트 생성하듯이 하려고했으나 자꾸 null값을 반환하여 왜그런지 찾아보니  제네릭 클래스는 컴포넌트화

01149.tistory.com

 

 

+ Recent posts