이번 강좌에서는 언리얼 오브젝트의 계층 구조를 다뤄보겠습니다.
먼저 게임인스턴스 헤더에 WebConnectionNew라는 변수를 같은 타입으로 동일하게 선언해줍시다.
////GameInstance.h
#pragma once
#include "Engine/GameInstance.h"
#include "WebConnection.h"
#include "ABGameInstance.generated.h"
UCLASS()
class ARENABATTLE_API UABGameInstance : public UGameInstance
{
GENERATED_BODY()
public:
UABGameInstance();
virtual void Init() override;
UPROPERTY()
class UWebConnection* WebConnection;
UPROPERTY()
class UWebConnection* WebConnectionNew;
};
////GameInstance.cpp
#include "ABCGameInstance.h"
#include "ABC.h"
UABCGameInstance::UABCGameInstance()
{
AB_LOG(Warning, TEXT("Constructor Call Start"));
WebConnection = CreateDefaultSubobject<UWebConnection>(TEXT("MyWebConnection"));
AB_LOG(Warning, TEXT("Constructor Call End"));
}
void UABCGameInstance::Init()
{
Super::Init();
AB_LOG_CALLONLY(Warning);
}
지난 번 CDO 강좌에서 WebConnection이라는 기본서브오브젝트(DefaultSubobject)를 생성해봤는데요,
CDO에서 기본서브오브젝트를 생성하면,
새로운 언리얼 오브젝트와 CDO 인스턴스 간에는 아래와 같은 관계를 가지게 됩니다.
아래는 이를 출력한 코드입니다.
/////ABGameInstance.cpp
void UABGameInstance::Init()
{
Super::Init();
AB_LOG_CALLONLY(Warning);
TArray<UObject*> DefaultSubobjects; //UObject 배열을 만들고,
GetDefaultSubobjects(DefaultSubobjects); //GetDefault 해서 언어온다.
for (const auto& Entry : DefaultSubobjects) //auto문을 이용해서 DefaultSubobjects 끝까지 순회한다
{
AB_LOG(Warning, TEXT("DefaultSubobject : %s"), *Entry->GetClass()->GetName());
AB_LOG(Warning, TEXT("Outer of DefaultSubobject : %s"), *Entry->GetOuter()->GetClass()->GetName());
}
}
이번에는 기본서브오브젝트와 동일한 타입의 WebConnection 언리얼 오브젝트를
게임 플레이 런타임인 Init 함수에서 생성하겠습니다.
게임플레이 런타임에서 언리얼 오브젝트를 생성하기 위해서는 NewObject 함수를 사용하면 됩니다.
이 때 첫 번째 인자 정보에 따라 새롭게 생성된 언리얼 오브젝트의 Outer가 결정됩니다.
이번 예제에서는 게임 인스턴스와의 연결을 위해 this를 넣어서 생성하겠습니다.
그러면 새롭게 생성되는 WebConnectionNew의 Outer는 게임 인스턴스가 됩니다.
////ABGameInstance.cpp
void UABGameInstance::Init()
{
Super::Init();
AB_LOG_CALLONLY(Warning);
TArray<UObject *> DefaultSubobjects;
GetDefaultSubobjects(DefaultSubobjects);
for (const auto& Entry : DefaultSubobjects)
{
AB_LOG(Warning, TEXT("DefaultSubobject : %s"), *Entry->GetClass()->GetName());
AB_LOG(Warning, TEXT("Outer of DefaultSubobject : %s"), *Entry->GetOuter()->GetClass()->GetName());
}
////추가
WebConnectionNew = NewObject<UWebConnection>(this);
AB_LOG(Warning, TEXT("Outer of NewObject : %s"), *WebConnectionNew->GetOuter()->GetClass()->GetName());
}
플레이를 누르면 런타임에서 최종 구조는 아래와 같이 되겠지요.
그러면 기존의 WebConnection과 새로운 WebConnectionNew와의 차이는 무엇일까요?
사실 사용에 있어서 두 객체간의 차이는 없다고 보면 됩니다.
다만 생성방식에서만 차이가 있는데,
전자인 WebConnection은 ABGameInstance의 인스턴스가 생성되면 자동으로 따라 생성되는 반면,
후자인 WebConnectionNew는 게임 플레이 런타임에서 우리가 직접 생성해줘야 합니다.
전자(WebConnection)의 방식을 스태틱(Static)하게 생성한다고 표현하겠습니다.
이는 기획상으로 변경될 여지가 거의 없는 오브젝트 묶음을 한번에 빠르게 생성할 때 유용합니다.
후자(WebConnectionNew)의 방식을 다이나믹(Dynamic)하게 생성한다고 표현하겠습니다.
후자 방식은 느리지만 세팅에 따라 다양한 오브젝트를 생성할 수 있어서,
유연하게 상황에 대처할 수 있다는 장점이 있습니다.
Level
퍼시스턴트 레벨 : 월드에는 최소 하나의 지정되는 레벨
스트리밍 레벨 : 월드에 무관하게 추가/삭제가 가능하도록 설계된 레벨
언리얼에서 게임 컨텐츠 구조는
ABGameInstance와 WebConnection간의 관계와 동일하게 부모-자식 관계가 확장된 구조를 따릅니다.
언리얼에서 모든 게임 컨텐츠의 기초은 월드에서부터 시작됩니다.
월드에는 최소 하나의 레벨이 지정되는데, 이를 퍼시스턴트 레벨(Persistent Level)이라고 합니다.
이 퍼시스턴트 레벨에는 추가로 월드의 초기 설정 값들을 지정한 월드세팅 언리얼 오브젝트가 있습니다.
기본 생성된 월드에는 레벨을 실시간으로 추가할 수 있는데,
월드에 무관하게 추가/삭제가 가능하도록 설계된 레벨을 스트리밍 레벨(Streaming Level)이라고 합니다.
이러한 레벨은 월드에서 독립적으로 행동하는 단위 오브젝트인 액터(Actor)의 묶음으로 구성되어 있습니다.
이 액터들은 다시 컴포넌트들로 구성되어 있지요.
지금까지 설명드린 월드의 계층 구성은 아래와 같이 정리할 수 있습니다.
이렇게 생성된 계층 구조는 설계에 따라
기본서브오브젝트 방식으로 관리할지, 런타임에서 실시간으로 생성해 관리할지 방식이 나뉘어지게 됩니다.
설명을 위해 월드 내에서 게임의 룰을 관장하는 액터인 게임모드를 가지고 예시를 들어보겠습니다.
FPS 게임을 제작할 때 레벨 디자이너가 레벨을 공들여서 제작했습니다.
이렇게 제작된 레벨에 플레이어들이 접속해 플레이를 할 때에는
데스매치(DeathMatch)룰을 적용할지,
깃발 뺏기(Capture the Flag) 룰을 적용할지를
방장이 정하도록 유연하게 적용할 수 있게 만드는 것이 좋은 디자인일 겁니다.
(레벨마다 변경이 될수 있으므로, 다이나믹인게 좋다?)
이 상황에서 월드세팅 오브젝트가
퍼시스턴트레벨 오브젝트에 스태틱 방식인 기본서브오브젝트로 포함되어 버리면, 월드세팅마다 레벨이 하나씩 만들어져야 합니다.
이러면 중복된 맵으로 인해 용량이 낭비되겠지요.
그래서 월드에서 월드세팅과 게임 모드는 후자인 다이나믹 방식으로 로딩되게 설계되어 있습니다.
반면에 액터는 설계자가 지정한 컴포넌트의 조합에 따라 월드에서 스스로 동작하도록 설계되어 있습니다.
그래서 액터는 스태틱하게 컴포넌트와 함께 생성하고 다 같이 소멸하는 것이 효과적입니다.
그래서 액터와 컴포넌트는 전자인 스태틱하게 로딩하도록 설계되어 있습니다.
아래는 이를 정리한 도식입니다.
FRangedWorld
이제 월드에 있는 언리얼 오브젝트들을 살펴봅시다.
월드에 속한 액터를 빠르게 검색하는 방법은 두 가지입니다.
하나는 For Ranged Loop 구조에 맞게 추가된 FRangedWorld를 사용하는 방법이 있습니다.
이 방식은 모든 액터를 간편하게 검색할 수 있다는 장점이 있습니다.
액터 내 컴포넌트는 모두 기본서브오브젝트로 등록되어 있으니 쉽게 검색이 가능합니다.
( "Runtime/Engine/Public/EngineUtils.h" 추가해야함)
////GameInstance.cpp
#include "Runtime/Engine/Public/EngineUtils.h"
void UABCGameInstance::Init()
{
....
UWorld* CurrentWorld = GetWorld();
for (const auto& Entry:FActorRange(CurrentWorld))
{
AB_LOG(Warning, TEXT("Actor : %s"), *Entry->GetName());
TArray<UObject*> Components;
Entry->GetDefaultSubobjects(Components);
for (const auto& CEntry : Components) //순회
{
AB_LOG(Warning, TEXT(" -- Component : %s"), *CEntry->GetName());
}
}
}
출력하면 아래와 같은 결과를 볼 수 있습니다.
여기서 액터 목록이 월드 아웃라이너와 다른데,
이는 게임인스턴스의 Init이 호출되는 시점에 아직 게임 플레이가 완성되지 않아서 그렇습니다.
검색만 간단한 방법?
TActorIterator
다른 하나는 TActorIterator를 사용하는 방식입니다.
이 방식은 액터 중에서도 우리가 원하는 타입만 선별해서 목록을 빠르게 뽑아낼 수 있어서
대부분 많이 사용하는 방식입니다.
아래는 현재 월드에서 스태틱메시액터 타입만 뽑아낸 예시입니다.
UWorld* CurrentWorld = GetWorld();
for (TActorIterator<AStaticMeshActor> It(CurrentWorld); It; ++It)
{
AB_LOG(Warning, TEXT("StaticMesh Actor : %s"), *It->GetName());
}
TObjectIterator
액터를 포함해 현재 월드에 로딩된 모든 언리얼 오브젝트를 가져오기 위해서는 TObjectIterator를 사용하면 됩니다.
아래 코드를 우리가 생성한 WebConnection가 세 개 로딩되어 있는 것을 확인할 수 있습니다.
하나는 ABGameInstance의 CDO에 있는 객체,
다른 하나는 실제 인스턴스의 기본서브오브젝트로 있는 객체,
나머지는 런타임에서 생성한 객체입니다.
(#include "Runtime/CoreUObject/Public/UObject/UObjectIterator.h" 추가해야 사용가능)
WebConnection->Host = TEXT("localhost");
WebConnectionNew->Host = TEXT("127.0.0.1");
for (TObjectIterator<UWebConnection> It; It; ++It)
{
UWebConnection* Conn = *It;
AB_LOG(Warning, TEXT("WebConnection Object Host : %s"), *Conn->Host);
}
이정도만 알아두면 이제 월드 내에서 원하는 언리얼 오브젝트를 쉽게 검색할 수 있을 겁니다~
액터에서 컴포넌트의 제작은 CreateDefaultSubobject를 사용해야 한다로 요약하면 되겠습니다.
'Unreal > Study' 카테고리의 다른 글
8. C++에서 블루프린트 확장 + 언리얼오브젝트 권한 + 프로퍼티 + AllowPrivateAccess (0) | 2019.05.28 |
---|---|
7. 액터의 제작+ 트랜스폼 + 블루프린트 액터, C++액터 (0) | 2019.05.28 |
5. UClass와 리플렉션 + 로그디버깅 (0) | 2019.05.27 |
4. 클래스 기본 객체 (CDO) + 로딩과정 + 로그디버깅 + 인스턴스생성 (0) | 2019.05.24 |
3. 언리얼 오브젝트 +구성 + 선언규칙 + 매크로 + 게임인스턴스 (0) | 2019.05.24 |