← 블로그 목록

몬스터 설계는 타입 데이터와 개체 상태를 분리할 때 훨씬 단단해진다

RPG 몬스터 설계의 첫 질문은 ‘이 정보가 모든 고블린에게 공통인가, 이 한 마리만의 상태인가’다. 플라이웨이트 패턴처럼 타입 데이터와 개체 상태를 분리하면 메모리 중복뿐 아니라 수정 범위까지 함께 좁아지고 데이터 의미도 선명해진다. 오브젝트 풀은 멋이 아니라 생성·해제 빈도가 실제 병목일 때만 얹는 최적화라는 점도 함께 정리한다.

몬스터 설계는 타입 데이터와 개체 상태를 분리할 때 훨씬 단단해진다

몬스터 설계는 타입 데이터와 개체 상태를 분리할 때 훨씬 단단해진다

RPG나 액션 게임에서 몬스터를 설계할 때 처음에는 모든 정보를 한 객체에 다 넣고 싶어진다. 이름, 공격력, 최대 체력, 위치, 현재 체력, 상태 이상, 드롭 테이블까지 한데 모아 두는 식이다. 처음에는 편해 보이지만 몬스터 수가 늘어나면 중복과 관리 비용이 금방 커진다.

이럴 때 가장 먼저 떠올려야 하는 질문은 “이 정보가 모든 고블린에게 공통인가, 아니면 이 고블린 한 마리만의 상태인가”다. 이 구분이 선명해지면 설계도 훨씬 단단해진다.

공통 데이터와 개별 상태를 나누면 메모리와 책임이 함께 정리된다

로버트 나이스트롬의 Game Programming Patterns는 플라이웨이트 패턴을 설명하면서, 많은 객체가 공유하는 상태와 각 인스턴스마다 다른 상태를 분리하라고 말한다. 숲 속 나무 예제에서 메시와 텍스처 같은 공통 정보는 공유하고, 위치와 크기 같은 정보만 각 나무 인스턴스에 둔다.

몬스터 설계도 비슷하다.

이렇게 나누면 고블린 100마리가 있어도 “고블린 기본 스펙”은 한 번만 들고 있으면 된다. 개별 몬스터는 그 타입을 참조하면서 자기 상태만 바꾸면 된다.

이 방식의 장점은 메모리 절약보다 수정 범위 축소에 있다

사람들이 이런 설계를 들으면 먼저 메모리 최적화를 떠올리지만, 실제 장점은 수정 범위가 줄어드는 데도 있다.

고블린 기본 공격력을 5에서 6으로 바꿔야 할 때, 공통 타입 데이터 한 곳만 바꾸면 된다. 반대로 모든 인스턴스가 기본값을 복사해 들고 있으면, 어떤 값이 원본이고 어떤 값이 변형된 값인지 점점 헷갈리기 쉽다.

즉 타입/인스턴스 분리는 성능 문제이면서 동시에 설계 책임 문제이기도 하다.

그렇다고 모든 것을 공유 객체로 빼면 오히려 불편해질 수 있다

플라이웨이트는 효율적인 패턴이지만, 무조건적으로 모든 값을 공유 객체로 빼는 것이 답은 아니다. 나이스트롬도 공통 상태를 공유하는 대신 포인터 추적과 캐시 미스 같은 비용을 고려해야 한다고 설명한다.

게임 몬스터로 치면 이런 식이다.

즉 핵심은 “어디까지 공유할 것인가”를 신중하게 정하는 것이다.

오브젝트 풀은 자주 생성하고 자주 버리는 경우에만 쓰는 편이 낫다

몬스터 설계를 이야기하다 보면 오브젝트 풀도 함께 나온다. Game Programming Patterns의 오브젝트 풀 장은, 비슷한 객체를 자주 만들고 자주 없애야 할 때 고정된 풀을 두고 재사용하면 성능과 메모리 단편화 문제를 줄일 수 있다고 설명한다.

이 패턴은 탄환, 파티클, 짧게 살아 있는 이펙트에 특히 잘 맞는다. 몬스터에도 쓸 수는 있지만 모든 경우에 필요한 것은 아니다.

즉 오브젝트 풀은 “멋있어 보여서” 넣는 최적화가 아니라, 생성/해제 빈도가 실제 병목일 때 쓰는 편이 맞다.

핵심 정리

게임 몬스터 설계에서 가장 먼저 해야 할 일은 공통 타입 데이터와 개체별 상태를 분리하는 것이다. 그렇게 해야 메모리 중복이 줄고, 데이터 의미가 선명해지고, 기본 스펙 수정도 쉬워진다.

여기에 오브젝트 풀은 필요할 때만 얹는 최적화다. 즉 좋은 몬스터 설계의 출발점은 모든 것을 객체 하나에 넣는 것이 아니라, 무엇이 공유 상태이고 무엇이 현재 상태인지를 구분하는 데 있다.

참고 자료

← 목록으로
Related

함께 읽으면 좋은 글

게임 개발프로그래밍리스프
고차 함수와 데이터 분리는 게임 로직의 변경 비용을 낮춘다

고차 함수와 데이터·로직 분리는 프로그램을 똑똑하게 만드는 마법이 아니라, 반복되는 패턴을 함수로 묶고 자주 바뀌는 설정을 코드 밖으로 빼서 변경 비용을 낮추는 추상화에 가깝다. SICP와 Unity ScriptableObject 사례를 함께 보면서, 게임처럼 규칙과 예외가 계속 늘어나는 환경에서 무엇을 코드로 두고 무엇을 데이터로 뺄지 가르는 기준을 정리한다.

게임 개발WinAPIWin32
WinAPI로 시작하는 게임 개발은 창 띄우기보다 메시지 루프를 이해하는 데서 시작된다

WinAPI로 게임 개발을 시작할 때 가장 먼저 이해해야 할 것은 창 만드는 함수가 아니라, 메시지를 받아 윈도우 프로시저로 보내는 Win32의 흐름 자체다. Win32와 Winsock은 자주 함께 등장하지만 역할이 다르므로 분리해서 익히는 편이 낫다. 입문자는 거대한 설계보다 작은 루프부터 직접 움직여 보는 순서를 정리한다.

게임 개발인공지능스타크래프트
스타크래프트 봇은 메모리 해킹보다 공식 API와 리플레이 분석부터 시작하는 편이 낫다

스타크래프트 봇 개발은 프로게이머 빌드를 코드로 옮기는 작업이라기보다, 공개 API로 게임 상태를 관측하고 그 위에서 행동을 선택하는 에이전트를 만드는 작업이다. BWAPI와 SC2 API, 리플레이 분석을 활용해 ‘관측 → 해석 → 행동’ 파이프라인을 만드는 입문 순서가 메모리 해킹보다 훨씬 현실적이라는 점을 정리한다.