서브컬처 게이머

세상의 모든 아름다운 것들을 위하여


나니노벨 Memory Management

개요

※이 글은 유니티 다이얼로그 시스템 에셋 ‘Naninovel(나니노벨)’의 한국어 번역 페이지입니다.

※모든 내용의 저작권 및 내용의 책임과 권한은 Naninovel에 있습니다.

※원문 페이지: (링크)

※마지막 수정일: 2024/12/14


일부 스크립트 명령어가 작동하려면 @bgm용 오디오 클립, @char용 캐릭터 프리팹 및/또는 표정 텍스처, @movie용 비디오 클립 등 리소스 로드가 필요합니다.

나니노벨은 대부분의 리소스를 사전에 로드 및 언로드합니다. 이에 관한 기본 동작은 리소스 공급자 구성에 있는 Resource Policy 설정을 기반으로 합니다.


Conservative 방식

균형 잡힌 메모리 활용을 제공하는 기본 모드입니다. 스크립트 실행에 필요한 모든 리소스는 재생을 시작할 때 미리 로드되고 스크립트 재생이 끝나면 언로드됩니다. @gosub 명령어가 참조하는 스크립트도 미리 로드됩니다. @goto 명령의 Hold 매개변수를 사용하여 추가 스크립트를 미리 로드할 수 있습니다.

다음은 Conservative 방식에 따라 리소스를 언로드/로드하는 방법에 대한 데모입니다.

Script1.nani

Script1, Script2 및 ScriptGosub의 리소스가 여기에 로드됩니다. Script2는 "@goto Hold!"로 탐색되었기 때문에 로드됩니다."@gosub"가 항상 미리 로드되어 있으므로 ScriptGosub가 로드됩니다.

...

gosub가 항상 미리 로드되어 있으므로 로딩 화면이 표시되지 않습니다.
@gosub ScriptGosub

...

'hold!'를 사용하고 있기 때문에 로딩 화면이 표시되지 않습니다.
@goto Script2 hold!

Script2.nani

Script1, Script2 및 ScriptGosub의 리소스는 모두 여전히 로드되어 있습니다.
왜냐하면 이 스크립트는 '@goto Hold!'를 사용하여 탐색되었기 때문입니다.
따라서 Script1의 종속성으로 간주됩니다.

...

'hold!'를 사용하지 않기 때문에 로딩 화면이 표시됩니다.
@goto Script3

Script3.nani

이제 Script1, Script2의 리소스가 언로드되고 리소스는
form Script3(이 스크립트)이 로드됩니다.
ScriptGosub의 리소스는 여기서 사용하고 있기 때문에 계속 로드됩니다.

...

gosub가 항상 미리 로드되어 있으므로 로딩 화면이 표시되지 않습니다.
@gosub ScriptGosub

...

'hold!'를 사용하지 않기 때문에 로딩 화면이 표시됩니다.
@goto Script4

Script4.nani

이제 모든 리소스가 언로드되었으며(ScriptGosub 포함)
Script4(이 스크립트) 형식의 리소스가 로드됩니다.

...

@stop

ScriptGosub.nani

여기서 탐색한 스크립트에 따라 다양한 리소스가 여기에 로드될 수 있습니다.

...

gosub는 항상 gosub로 이동하는 스크립트와 함께 로드되고 스크립트가 언로드될 때까지 언로드되지 않으므로 로딩 화면이 표시되지 않습니다.
@return

Optimistic 방식

재생된 스크립트에 필요한 모든 리소스와 @goto 및 @gosub 명령에 지정된 모든 스크립트의 모든 리소스는 미리 로드되며 @goto 명령에 release 파라미터가 지정되지 않는 한 언로드되지 않습니다.

이는 화면 로딩을 최소화하고 원활한 롤백을 허용하지만 리소스를 언로드할 시기를 수동으로 지정해야 하므로 모바일 기기나 웹 브라우저와 같이 엄격한 메모리 제한이 있는 플랫폼에서 메모리 부족 예외상황이 발생할 위험이 증가합니다.

아래는 위 Conservative 방식과 유사한 스크립트 세트(데모)이지만 Optimistic 방식의 경우를 설명합니다.

Script1.nani

Script1, Script2, Script3 및 ScriptGosub의 리소스가 모두 여기에 로드됩니다.
Script4는 '@goto release!'로 탐색되었기 때문에 로드되지 않습니다.

...

gosub가 항상 미리 로드되어 있으므로 로딩 화면이 표시되지 않습니다.
@gosub ScriptGosub

...

'릴리스!'가 아닌 이상 로딩 화면은 기본적으로 표시되지 않습니다. 지정됩니다.
@goto Script2

Script2.nani

Script4를 제외한 모든 항목은 여전히 ​​로드됩니다.

...

'릴리스!'가 아닌 이상 로딩 화면은 기본적으로 표시되지 않습니다. 지정됩니다.
@goto Script3

Script3.nani

Script4를 제외한 모든 항목은 여전히 ​​로드됩니다.

...

gosub가 항상 미리 로드되어 있으므로 로딩 화면이 표시되지 않습니다.
@gosub ScriptGosub

...

이제 '릴리스!'로 인해 로딩 화면이 표시됩니다.
@goto Script4 release!

Script4.nani

여기서 탐색했기 때문에 Script4를 제외한 모든 리소스가 이제 언로드됩니다.

...

@stop

ScriptGosub.nani

여기서 탐색한 스크립트에 따라 다양한 리소스가 여기에 로드될 수 있습니다.

...

gosub는 항상 gosub로 이동하는 스크립트와 함께 로드되고 스크립트가 언로드될 때까지 언로드되지 않으므로 로딩 화면이 표시되지 않습니다.
@return

리소스 정책 선택하기

일반적으로 기본 옵션인 ‘Conservative’ 정책을 고수하는 것이 좋습니다. 이는 모든 대상 플랫폼에 적합한 균형 잡힌 메모리 사용을 제공하는 동시에, 필요 시 hold! 플래그로 유연한 스크립트 병합 기능을 제공하기 때문입니다.

그러나 스탠드얼론이나 게임용 콘솔기기와 같이 더 많은 램을 제공하는 더 강력한 플랫폼 독점으로 하는 경우에는 ‘Optimistic’ 방식을 선택하여 메모리의 리소스에 대한 큰 덩어리를 유지하고 로딩 화면을 최소화할 수 있습니다.

대안이 될 만한 방식으로써 또 다른 시나리오는, 나니노벨이 커스텀 게임 루프 내에서 다이얼로그 시스템으로써 사용되는 경우입니다. 그러한 경우 자체 리소스 관리 시스템을 갖게 될 가능성이 높습니다. ‘Optimistic’ 방식을 선택해도 기본적으로 스크립트를 재생하기 전에 필요한 모든 리소스를 로드된 상태로 유지하는 것 외에는 아무것도 하지 않으므로 명시적으로 release! 플래그를 사용하지 않는 한 문제가 되지 않습니다.

다음은 리소스 정책의 차이점을 표로 요약한 것입니다.

리소스 정책메모리 사용량씬 로딩 빈도롤백 차이
Conservative균형잡힘잦음hold 내에서는 빠름
Optimistic높음거의 없음release 외에는 빠름

액터 리소스

액터(캐릭터, 배경, 텍스트 출력기 및 선택지 처리기)는 Naninovel의 핵심 실체(엔터티)입니다. 액터가 사용하는 대부분의 메모리는 외양(배리에이션 등)과 연관되어 있습니다.

외양

일부 액터 구현에는 외양과 리소스에 1:1로 매핑됩니다. 스프라이트 액터 외양은 단일 텍스처 에셋과 연결되고, 비디오 액터 외양은 단일 비디오 클립과 매핑되는 등입니다. 이를 통해 나니노벨은 시나리오 스크립트에서 참조된 특정 외양을 기반으로 리소스를 관리할 수 있습니다.

예를 들어, 주어진 스크립트에서 스프라이트 캐릭터의 Happy 및 Sad 외양만 사용되는 경우 캐릭터의 다른 외양의 수에 관계없이 스크립트가 재생되기 전에 Happy.png 및 Sad.png 텍스처만 미리 로드됩니다.

그러나 레이어드 방식, 다이스 스프라이트, 제네릭, Live2D 및 Spine 액터에서는 관련 외양 묘사를 위해서는 모노리스 프리팹이 필요하므로 스프라이트 방식과 달리 리소스를 독립적으로 로드할 수 없습니다. 이러한 경우 나니노벨은 모든 종속성과 함께 전체 프리팹을 미리 로드하게 되는데, 어떤 외양이 사용되는지에 관계없이 액터가 어떤 명령어에서도 참조되지 않을 때만 언로드합니다.

액터 언로딩

나니노벨은 기본적으로 스크립트 리소스를 언로드할 때 사용하지 않는 액터를 자동으로 제거하고 관련 게임 오브젝트를 삭제(언로드)합니다. 액터를 수동으로 삭제하려면 리소스 공급자 구성 메뉴에서 Remove Actors 옵션을 비활성화하고 @remove 명령어를 사용하세요.

@back id:LayeredBackground
@char GenericCharacter
@char DicedCharacter

; 'Remove Actors'가 비활성화되면 'NextScript'가 로드될 때
; 'LayeredBackground'가 삭제되지 않지만 두 캐릭터는 모두 삭제됩니다.

@hide GenericCharacter,DicedCharacter wait!
@remove GenericCharacter,DicedCharacter
@goto NextScript

또는 ‘*’ 파라미터와 함께 @remove를 사용하여 기존 액터(텍스트 출력기 및 선택지 처리기 포함)를 모두 삭제하거나 파라미터만 포함하여 @resetState를 사용하여 특정 유형의 액터를 즉시 삭제합니다.

캐릭터의 경우 ICharacterManager, 배경의 경우 IBackgroundManager를 사용하면 됩니다.

...
@goto NextScript
; 모든 존재하는 배경 리소스 언로드하기
@resetState only:IBackgroundManager

유지시간 관리

리소스 공급자 관리자는 로드된 리소스에 대한 참조를 추적하고 사용자(‘보유자’)가 리소스를 사용(‘보유’)하지 않을 때 리소스를 삭제(언로드)합니다.

이 메커니즘은 스크립트 명령어에서 가장 두드러집니다.

예를 들어 커스텀 명령어를 사용하여 배경 음악을 재생한다고 가정해 보겠습니다. 오디오 플레이어를 재생하려면 오디오 클립 에셋(리소스)이 필요하므로 명령어가 실행되기 전에 에셋을 미리 로드하고 ‘hold’하고 이후에 해제해야 합니다.

public class PlayMusic : Command, Command.IPreloadable
{
    public StringParameter MusicName;

    private IAudioManager audio => Engine.GetService<IAudioManager>();

    public async UniTask PreloadResources ()
    {
        await audio.AudioLoader.LoadAndHold(MusicName, this);
    }

    public void ReleasePreloadedResources ()
    {
        audio.AudioLoader.Release(MusicName, this);
    }

    public override async UniTask Execute (AsyncToken asyncToken = default)
    {
        await audio.PlayBgm(MusicName, asyncToken: asyncToken);
    }
}

이 명령어는 Command.IPreloadable 인터페이스를 구현합니다. 스크립트 플레이어는 이러한 명령어를 감지하고 사전 로드 및 언로드 메서드를 호출하여 명령이 실행되기 전에 에셋이 준비되고 실행 이후에 해제되는지 확인합니다.


리소스 공유

어떤 경우에는 나니노벨과 커스텀 게임 플레이 모드 간에 리소스를 공유하고 싶을 수도 있습니다. 커스텀 게임플레이가 나니노벨과 독립적으로 구현되는 경우(커스텀 모드가 활성화되면 엔진이 비활성화됨) 아무런 문제가 없어야 합니다. 하지만 커스텀 모드와 나니노벨을 동시에 사용하는 경우에는 리소스 사용 방식에 주의를 기울여야 합니다.

예를 들어 일부 UI 요소의 소스로도 사용되는 외양 텍스처가 있는 나니노벨의 스프라이트 배경이 있다고 가정해 보겠습니다. 어느 시점에서 나니노벨은 텍스처 해제를 시도하고 UI 요소에서도 사라질 것입니다. 엔진이 텍스처를 사용하고 있고 언로드되어서는 안 된다는 사실을 인식하지 못하기 때문에 이런 일이 발생합니다.

에셋을 사용하고 있음을 나니노벨에 알리려면 리소스 공급자 서비스의 Hold 메소드를 사용하십시오.

var resourceManager = Engine.GetService<IResourceProviderManager>();
resourceManager.Hold(asset, holder);

에셋을 보유하고 있는 동안 나니노벨에 의해 언로드되지 않으므로 메모리 누수를 방지하기 위해 에셋을 폐기하는 것은 당신 몫입니다.

var holdersCount = resourceManager.Release(asset, holder);
// 에셋을 보유하고 있는 홀더가 아무도 없는 경우 에셋을 언로드해야 합니다.
if (holdersCount == 0) Resources.UnloadAsset(asset);

‘Holder’는 모든 객체에 대한 참조가 될 수 있습니다. 일반적으로 에셋을 사용하는 클래스와 동일합니다. 이는 홀더를 구별하고 동일한 홀더가 실수로 리소스를 여러 번 보유하는 것을 방지하는 데 사용됩니다.

다음은 나니노벨이 에셋을 언로드하는 것을 방지하는 Unity 구성 요소의 예입니다.

using Naninovel;
using UnityEngine;

public class HoldObject : MonoBehaviour
{
    public Object ObjectToHold;

    private async void Start()
    {
        while (!Engine.Initialized) await UniTask.DelayFrame(1);
        Engine.GetService<IResourceProviderManager>().Hold(ObjectToHold, this);
    }
}

연관글 목록

답글 남기기

이메일 주소는 공개되지 않습니다. 필수 필드는 *로 표시됩니다