유니크포인터 이동시멘틱 : MoveTemp
공유포인터 생성 : TShard<class> Name = MakeShareable()
마지막으로 정리한 내용은 언리얼 엔진에서 일반 C++ 객체를 다룰 때 편리하게,
사용할 수 있는 스마트포인터 라이브러리에 관한 내용입니다.
로또에 당첨되 돈이 넘쳐나게된 당신은 드디어 자신만의 집을 짓게 됩니다.
일반 C++ 클래스인 집을 아래와 같이 선언하겠습니다. 이전 강좌에서 언급했지만,
언리얼 엔진 코드에서는 일반 C++ 클래스를 선언할 때 F 접두사를 씁니다.
집에 대한 클래스 선언과 집 크기를 멤버 변수로 하나 만들었습니다.
//ABGameInstance.h
class FHouse
{
public:
int32 Size = 10;
};
그런데 한 집에서 오래 살다보니 보다 넓은 집을 짓고 싶어집니다.
그럴러면 건물을 허물고 다시 지어야 겠지요?
집을 짓고 허무는 과정을 일반 C++ 코드로 표현하면 다음과 같습니다.
//ABGameInstance.cpp
void UABGameInstance::Init()
{
Super::Init();
// 1단계
AB_LOG(Warning, TEXT("****** 1단계 ******"));
FHouse* NewHouseAddress = new FHouse(); //동적할당
AB_LOG(Warning, TEXT("집을 새로 지었습니다. 내집크기 : %d"), NewHouseAddress->Size);
if (NewHouseAddress)
{
delete NewHouseAddress; //삭제
AB_LOG(Warning, TEXT("내가 직접 집을 철거했습니다. 집크기 : %d"), NewHouseAddress->Size);
NewHouseAddress = nullptr;
}
}
집 오브젝트가 소멸되었는지의 판단은 집 크기 값을 참조했을 때,
아래 로그와 같이 음수의 쓰레기 값이 들어있는지로 판단하겠습니다. 아래와 같이 말이지요.
그런데 (코딩을 하다보면) 새로운 집을 지을 때 깜빡하고 이전 집을 철거안하지 않은 채로 새로운 곳에 집을 짓는 경우가 종종 발생합니다. (해제 및 삭제를 안하는 경우)
이렇게 철거가 안된 집들이 계속 늘어나게 되면 언젠가는 새로운 집을 지을 수 있는 공간이 부족하게 되 사회적인 문제로 발전하게 됩니다. 이러한 실수를 보완하기 위해서 스마트 포인터개념이 등장했습니다. 스마트포인터는 내가 직접 철거하는 대신, "나 떠납니다"라고 신고만하면 알아서 집을 철거해주는 편리한 대행 서비스입니다.
스마트 포인터
스마트 포인터의 사용법은 일반 포인터와 거의 동일합니다.
하지만 일반 포인터와 스마트 포인터가 지양하는 컨셉은 좀 다릅니다.
일반 포인터가 객체에 접근가능한 주소를 보관하는 개념이라면, 스마트 포인터는 객체의 소유권을 증명하는 개념입니다.
원래 C++언어에서는 집주소만 보관할 수 있어서 집을 없애려면 그 주소(포인터)에 내가 직접 가서 철거를 했어야 했습니다. 하지만 스마트 포인터가 등장하면서 집의 소유권이 없으면 그 집은 자동으로 철거되버립니다.(자동으로 삭제 delete를 해줄 필요가 없다)
이제는 집주인이 깜빡해도 집주인이 떠난 집은 깔끔하게 정리되면서 토지의 재활용성이 높아지게 되었습니다.
이 편리한 서비스가 개발자들 사이에서 점점 보편화되다보니 C++ 11 버젼에 이르러서는,
공식적인 표준으로 인정받게 됩니다.
C++에서 대표적인 스마트 포인터 서비스는 용도에 따라 삼총사가 있습니다.
언리얼에서는 이를 T로 시작하는 라이브러리로 제공합니다. 이번 강좌에서는 이들에 대해 알아보겠습니다.
스마트 포인터 종류
1. 유니크포인터(TUniquePtr) : 집의 유일한 소유권을 보장하는 서비스입니다. 공동 소유는 불가합니다.
2. 공유포인터(TSharedPtr) : 집에 대해 공동 소유권을 주장할 수 있는 서비스입니다. (참조 횟수 카운트 가능)
3. 약포인터(TWeakPtr) : 집에 대한 정보는 가지고 있으나, 소유권은 없는 서비스입니다.
굳이 예를 들자면 부동산 중개업이라고 보면 됩니다.
1. 유니크포인터(TUniquePtr)
스마트 포인터를 사용하게 되면 포인터처럼 사용하지만 집주소가 아닌 집의 소유를 증명할 수 있는 집문서의 개념으로 변화됩니다. 스마트포인터는 집의 소유를 증명하는 집문서가 모두 사라지면 자동으로 집은 철거되는 개념이라고 설명드렸습니다.
여러분들이 집을 지을 때, 단독 명의로 집을 짓고 싶다면 유니크 포인터 서비스를 이용하면 됩니다.
단독 명의로 선언하면 이 집을 혼자서 독차지할 수 있지만 다른 사람들과 나누어 가질 수는 없습니다.
하지만 혼자 소유한다고 해서 매매가 불가능한 것은 아닙니다.
이미 만들어진 집은 그대로 두고 주인만 바꿔치기할 수 있습니다. (std::move) ,MoveTemp
이러한 개념을 C++에서는 무브시멘틱(Move Semantic) 이라고 하는데,
언리얼 엔진에서는 MoveTemp라는 이름으로 API를 제공합니다.
아래는 위에서 언급한 모든 기능을 구현한 코드입니다.
1단계와 2단계의 코드를 잘 비교해보시기 바랍니다.
void UABGameInstance::Init()
{
Super::Init();
// 1단계
AB_LOG(Warning, TEXT("****** 1단계 ******"));
FHouse* NewHouseAddress = new FHouse();
AB_LOG(Warning, TEXT("집을 새로 지었습니다. 내집크기 : %d"), NewHouseAddress->Size);
if (NewHouseAddress)
{
delete NewHouseAddress;
AB_LOG(Warning, TEXT("내가 직접 집을 철거했습니다. 집크기 : %d"), NewHouseAddress->Size);
NewHouseAddress = nullptr;
}
// 2단계
AB_LOG(Warning, TEXT("****** 2단계 ******"));
NewHouseAddress = new FHouse();
NewHouseAddress->Size = 100;
AB_LOG(Warning, TEXT("집을 다시 지었습니다. 집크기 : %d"), NewHouseAddress->Size);
{
TUniquePtr<FHouse> MyHouseDeed = TUniquePtr<FHouse>(NewHouseAddress);
if (MyHouseDeed.IsValid())
{
AB_LOG(Warning, TEXT("이 집은 제 단독소유 주택입니다. 내집크기 : %d"), MyHouseDeed->Size);
}
//TUniquePtr<FHouse> FriendsHouseDeed = MyHouseDeed; // 컴파일 에러! 단독 소유만 가능
TUniquePtr<FHouse> FriendsHouseDeed = MoveTemp(MyHouseDeed); // 집은 그대로 두고 집주인만 변경
if (!MyHouseDeed.IsValid())
{
AB_LOG(Warning, TEXT("친구에게 집을 팔았습니다. 친구집크기 : %d"), FriendsHouseDeed->Size);
}
}
AB_LOG(Warning, TEXT("집문서가 사라져서 집은 자동으로 철거되었습니다. 집크기 : %d"), NewHouseAddress->Size);
}
//Movetemp
template <typename T>
FORCEINLINE typename TRemoveReference<T>::Type&& MoveTemp(T&& Obj)
{
typedef typename TRemoveReference<T>::Type CastType;
// Validate that we're not being passed an rvalue or a const object - the former is redundant, the latter is almost certainly a mistake
static_assert(TIsLValueReferenceType<T>::Value, "MoveTemp called on an rvalue");
static_assert(!TAreTypesEqual<CastType&, const CastType&>::Value, "MoveTemp called on a const object");
return (CastType&&)Obj;
}
2. 공유포인터(TSharedPtr)
여러분이 제작한 객체를 여러 군데에서 사용하고자 할 경우에는,
즉 공동명의로 집을 소유하고 싶으면 공유 포인터 서비스를 이용하면 됩니다.
공유포인터는 내부적으로 레퍼런스 카운팅 기법을 사용합니다. MakeShareable()
누군가 객체를 참조할 때마다 레퍼런스 카운트가 하나씩 올라가고, 해지할 때 하나씩 내려갑니다.
이렇게 관리하다가 레퍼런스 카운트가 0이되면 스마트포인터는 자동으로 해당 객체를 소멸시킵니다.
아래는 공유포인터를 사용해 구현한 예시입니다. 이제 집문서를 복제해서 아무한테나 마구 뿌릴 수 있게 되었습니다. 하지만 세상에 공짜는 없습니다. 이제 여러분은 집을 누구에게나 공유할 수 있게 되었지만, 집을 철거할 때에는 뿌린 집문서를 모두 거두어서 소멸시켜야 합니다.
////선인 및 초기화
// Create an empty shared pointer
TSharedPtr<FMyObjectType> EmptyPointer;
// Create a shared pointer to a new object
TSharedPtr<FMyObjectType> NewPointer(new FMyObjectType());
// Create a Shared Pointer from a Shared Reference
TSharedRef<FMyObjectType> NewReference(new FMyObjectType());
TSharedPtr<FMyObjectType> PointerFromReference = NewReference;
// Create a Thread-safe Shared Pointer
TSharedPtr<FMyObjectType, ESPMode::ThreadSafe> NewThreadsafePointer = MakeShared<FMyObjectType, ESPMode::ThreadSafe>(MyArgs);
//Reset 기능
PointerOne.Reset();
PointerTwo = nullptr;
// Both PointerOne and PointerTwo now reference nullptr.
AB_LOG(Warning, TEXT("****** 3단계 ******"));
//포인터1
NewHouseAddress = new FHouse();
NewHouseAddress->Size = 150.0f;
AB_LOG(Warning, TEXT("집을 또 다시 지었습니다. 집크기 : %d"), NewHouseAddress->Size);
{
//참조포인터1 생성 (포인터1을 이용해서)
TSharedPtr<FHouse> MyHouseDeed = MakeShareable(NewHouseAddress); // 만들어진 집을 차후에 등록
if (MyHouseDeed.IsValid())
{
AB_LOG(Warning, TEXT("공동 소유 가능한 집이 되었습니다. 내집크기 : %d"), MyHouseDeed->Size);
if (MyHouseDeed.IsUnique())
{
AB_LOG(Warning, TEXT("현재는 혼자 소유하고 있습니다. 내집크기 : %d"), MyHouseDeed->Size);
}
}
//참조포인터2 생성(참조 포인터1을 이용)
TSharedPtr<FHouse> FriendsHouseDeed = MyHouseDeed;
if (!FriendsHouseDeed.IsUnique())
{
AB_LOG(Warning, TEXT("친구와 집을 나눠가지게 되었습니다. 친구집크기 : %d"), FriendsHouseDeed->Size);
}
MyHouseDeed.Reset(); // 내가 집 소유권을 포기함
if (FriendsHouseDeed.IsUnique())
{
AB_LOG(Warning, TEXT("이제 친구만 집을 소유하고 있습니다. 친구집크기 : %d"), FriendsHouseDeed->Size);
}
AB_LOG(Warning, TEXT("집은 아직 그대로 있습니다. 집크기 : %d"), NewHouseAddress->Size);
}
AB_LOG(Warning, TEXT("집은 자동 철거되었습니다. 집크기 : %d"), NewHouseAddress->Size);
아래는 공유 포인터를 사용한 결과입니다. 모든 공유 포인터가 소멸되어야 객체가 소멸되는 것을 확인할 수 있습니다.
공유 포인터 문제점
공유포인터는 가장 많이 사용하는 스마트포인터라고 할 수 있습니다.
하지만 뿌린 집문서가 모두 사라져야 집이 소멸되는 특징이 있기 때문에, 예상치 못하는 상황이 발생할 수 있습니다.
대표적인 문제가 공유 포인터의 순환 참조(Circular Reference) 문제입니다.
이 문제를 직접 확인하고 테스트하기 위해, 집 클래스를 확장해 다른 집 문서를 보관하는 기능을 추가합시다.
class FHouse
{
public:
TSharedPtr<FHouse> OthersDeed; //추가
int32 Size = 10;
};
아래 코드처럼 나와 친구가 각각 집을 짓고, 서로 집을 공동 소유하는 문서를 보관하게 만들어봅시다.
둘은 서로의 집에 대해 소유권을 별도로 가지고 있기 때문에, 양쪽의 집문서가 모두 소멸되도 집이 철거되지 않고 그대로 남아있는 문제가 발생합니다
NewHouseAddress = new FHouse();
NewHouseAddress->Size = 200.0f;
AB_LOG(Warning, TEXT("집을 한번 더 다시 지었습니다. 첫번째집크기 : %d"), NewHouseAddress->Size);
FHouse* NewHouseAddress2 = new FHouse();
NewHouseAddress2->Size = 250.0f;
AB_LOG(Warning, TEXT("친구도 집을 직접 지었습니다. 두번째집크기 : %d"), NewHouseAddress2->Size);
{
TSharedPtr<FHouse> MyHouseDeed = MakeShareable(NewHouseAddress);
AB_LOG(Warning, TEXT("내 집은 내가 소유합니다. 내집크기 : %d"), MyHouseDeed->Size);
TSharedPtr<FHouse> FriendsHouseDeed = MakeShareable(NewHouseAddress2);
AB_LOG(Warning, TEXT("친구 집은 친구가 소유합니다. 친구집크기 : %d"), FriendsHouseDeed->Size);
MyHouseDeed->OthersDeed = FriendsHouseDeed;
AB_LOG(Warning, TEXT("친구 집을 공동 소유하고 문서를 내 집에 보관합니다. 친구집크기 : %d"), MyHouseDeed->OthersDeed->Size);
FriendsHouseDeed->OthersDeed = MyHouseDeed;
AB_LOG(Warning, TEXT("친구도 내 집을 공동 소유하고 문서를 자기 집에 보관합니다. 내집크기 : %d"), FriendsHouseDeed->OthersDeed->Size);
}
AB_LOG(Warning, TEXT("집문서가 사라져도 내가 지은 집이 자동 철거되지 않습니다. 첫번째집크기 : %d"), NewHouseAddress->Size);
AB_LOG(Warning, TEXT("친구가 지은 집도 자동 철거되지 않습니다. 두번째집크기 : %d"), NewHouseAddress2->Size);
NewHouseAddress->OthersDeed.Reset();
AB_LOG(Warning, TEXT("친구가 지은 집을 수동으로 철거했습니다. 집주소가 남아있어서 다행입니다. 두번째집크기 : %d"), NewHouseAddress2->Size);
AB_LOG(Warning, TEXT("이제서야 내가 지은 집도 자동 철거됩니다. 첫번째집크기 : %d"), NewHouseAddress->Size);
아래는 위의 코드를 실행한 결과화면입니다.
이전에 저장해 둔 포인터 변수가 있어서 다행히 직접 삭제를 진행해 메모리 릭을 막았습니다.
하지만 이와 같이 공유포인터에만 모든 것을 의존하면 문제가 발생할 수 있음을 확인할 수 있습니다.
3. 약포인터(TWeakPtr)
공유 포인터에 의한 문제를 해결하기 위해 등장한 것이 약포인터입니다.
약포인터는 부동산 중개업자로 비유할 수 있습니다. 집에 대한 소유권은 없지만 집에 대해서는 잘 알고 있죠.
위의 문제를 해결하기 위해 한 쪽에는 약포인터로 선언해 필요할 때에만 중개업자를 통해서 이용하도록(?) 구성해봅시다. 그러면 양쪽이 모두 서로의 집을 사용할 수 있으면서,
양쪽 집문서가 소멸되면 자동으로 집도 철거되게 만들 수 있습니다.
먼저 약포인터 멤버 변수를 추가합시다.
class FHouse
{
public:
TSharedPtr<FHouse> OthersDeed;
TWeakPtr<FHouse> AccessHouse;
int32 Size = 10;
};
한쪽은 공유포인트를 쓰고, 다른 한쪽은 약포인터를 사용해서 상호 참조하게 만들어줍시다.
공유포인터를 사용하는 쪽은 편하게 쓸 수 있는 반면,
약포인터를 쓰는 곳은 약포인터가 제공하는 Pin()함수를 사용해서 항상 집문서가 제대로 되었는지 체크해주어야 하는 불편함이 있습니다.
하지만 이렇게 약포인터를 사용하면 위의 문제 없이 안전하게 사용할 수 있습니다.
// 5단계
AB_LOG(Warning, TEXT("****** 5단계 ******"));
NewHouseAddress = new FHouse();
NewHouseAddress->Size = 300.0f;
AB_LOG(Warning, TEXT("이제 마지막으로 집을 짓습니다. 첫번째집크기 : %d"), NewHouseAddress->Size);
NewHouseAddress2 = new FHouse();
NewHouseAddress2->Size = 350.0f;
AB_LOG(Warning, TEXT("친구도 집을 다시 지었습니다. 두번째집크기 : %d"), NewHouseAddress2->Size);
{
TSharedPtr<FHouse> MyHouseDeed = MakeShareable(NewHouseAddress);
AB_LOG(Warning, TEXT("내 집은 내가 소유합니다. 내집크기 : %d"), MyHouseDeed->Size);
TSharedPtr<FHouse> FriendsHouseDeed = MakeShareable(NewHouseAddress2);
AB_LOG(Warning, TEXT("친구 집은 친구가 소유합니다. 친구집크기 : %d"), FriendsHouseDeed->Size);
MyHouseDeed->OthersDeed = FriendsHouseDeed;
AB_LOG(Warning, TEXT("친구 집을 공동 소유하고 문서를 내 집에 보관합니다. 친구집크기 : %d"), MyHouseDeed->OthersDeed->Size);
FriendsHouseDeed->AccessHouse = MyHouseDeed;
AB_LOG(Warning, TEXT("친구가 내 집 정보를 열람합니다. 내집크기 : %d"), FriendsHouseDeed->AccessHouse.Pin()->Size);
}
AB_LOG(Warning, TEXT("내가 지은 집은 자동 철거됩니다. 첫번째집크기 : %d"), NewHouseAddress->Size);
AB_LOG(Warning, TEXT("친구가 지은 집도 자동 철거됩니다. 두번째집크기 : %d"), NewHouseAddress2->Size);
아래는 이를 실행한 로그 결과입니다.
언리얼 오브젝트의 메모리 관리
위에서 언급한 스마트 포인터 라이브러리는 일반 C++ 객체를 위한 라이브러리이고,
언리얼 오브젝트에는 사용할 수 없습니다.
언리얼 오브젝트는 언리얼 엔진 가상머신의 가비지 컬렉션(Garbage Collection, 줄여서 GC) 시스템에 의해 자동으로 관리되기 때문입니다.
하나의 언리얼 오브젝트가 GC 시스템에 의해 자동 관리되기 위해서는
선언에 반드시 UPROPERY 매크로가 들어가야 합니다.
그러면 언리얼 엔진에 의해서 필요가 없어질 때 자동으로 회수됩니다.
GC 시스템이 언리얼 오브젝트를 관리하는 방식은
스마트 포인터 삼총사 중 두 번째에 설명한 공유 포인터와 거의 유사합니다.
다만 다른 점은 중앙의 GC 시스템에 의해서 모든 관리되기 때문에,
메모리에서 해지되는 타이밍을 정확히 예측할 수 없다는 점입니다.
따라서 언리얼 오브젝트의 포인터를 소멸할 때에는 BeginConditionalDestroy()라는 함수를 호출해주고,
시스템이 해지해줄 때까지 기다리는 수 밖에 없습니다.
(액터의 경우에는 월드의 씬 정보를 업데이트 해야하기 때문에, 먼저 DestroyActor 함수도 추가로 호출해야 합니다. )
가비지 컬렉션은 프로젝트 세팅에서 지정된 시간마다 언리얼 오브젝트를 감지해서 제거해줍니다.
이는 프로젝트 세팅의 Garbage Collection 탭에서 확인할 수 있습니다.
빠른 테스트를 위해서 기본 값 60초를 변경해 10초마다 거둬들이도록 조정했습니다.
이제 언리얼 오브젝트가 잘 회수되는지 테스트 코드로 살펴봅시다.
테스트를 위해 ABGameInstance의 선언에 있는WebConnectionNew 멤버 변수의 UPROPERTY를 주석처리했습니다.
UPROPERTY매크로를 선언하면 게임 인스턴스가 제거될 때까지,
게임인스턴스는 언리얼 오브젝트의 인스턴스를 계속 유지하기 때문입니다.
앞서서 언리얼 오브젝트는 해지 명령을 내려도 바로 해지되지 않고 GC 시스템에 의해 회수된다고 설명드렸습니다.
이를 테스트하기 위해 1초마다 주기적으로 콜백함수를 호출하는 타이머핸들과 타이머콜백함수를 함께 선언했습니다
아래는 새롭게 수정된 ABGameInstance의 선언입니다.
//GameInstance.h
UCLASS()
class ABC_API UABGameInstance : public UGameInstance
{
GENERATED_BODY()
public:
UABCGameInstance();
virtual void Init() override;
UPROPERTY(BlueprintReadOnly, VisibleAnywhere, Category = "WebService")
class UWebConnection* WebConnection;
//UPROPERTY()
class UWebConnection* WebConnectionNew;
UPROPERTY()
FStreamableManager AssetLoader;
UFUNCTION()
void RequestTokenComplete(const FString& Token);
FTimerHandle ObjectCheckTimer;
UFUNCTION()
void CheckUObjectAlive();
};
이제 WebConnection 언리얼 오브젝트를 생성해 WebConnectionNew 변수에 할당하고,
ConditionalBeginDestroy() 함수를 실행해 제거 명령을 내립시다.
삭제 명령을 내리자마자 타이머를 돌려서 GC 시스템에 의해 삭제될 때까지 로그를 찍도록 타이머를 설정합시다.
//ABGameInstance.cpp
// 6단계
AB_LOG(Warning, TEXT("****** 6단계 ******"));
WebConnectionNew = NewObject<UWebConnection>();
WebConnectionNew->Host = TEXT("127.0.0.1");
//WebConnectionNew->AddToRoot();
WebConnectionNew->ConditionalBeginDestroy()
//GetWorld()->ForceGarbageCollection(true);
GetWorld()->GetTimerManager().SetTimer(ObjectCheckTimer, this, &UABCGameInstance::CheckUObjectAlive, 1.0f, true);
}
void UABGameInstance::CheckUObjectAlive()
{
if (!WebConnectionNew)
{
AB_LOG(Warning, TEXT("언리얼오브젝트가 널입니다."));
return;
}
if (!WebConnectionNew->IsValidLowLevel())
{
AB_LOG(Warning, TEXT("언리얼오브젝트가 유효하지 않습니다."));
return;
}
AB_LOG(Warning, TEXT("언리얼오브젝트가 아직 살아있습니다. : %s"), *WebConnectionNew->Host);
}
참고로 주석처리한 월드의 ForceGarbageCollection(true) 함수를 실행해 GC 시스템에게 바로 자원 회수를 명령할 수 있습니다.
이는 GC 시스템에 의해 많은 수의 언리얼 오브젝트를 한번에 회수할 때,
일시적인 랙이 생기는 딸꾹질(Hiccup) 현상을 미연에 방지해 줄 수 있는 효과가 있습니다.
반대로 언리얼 오브젝트의 삭제를 명령했음에도 불구하고,
계속 언리얼 오브젝트를 메모리에 유지시키고 싶을 떄에는 주석처리한 AddToRoot() 함수를 사용하면
자원회수를 원천적으로 봉쇄하는 것도 가능합니다
언리얼 오브젝트의 약포인터
언리얼 오브젝트 메모리 관리는 공유 포인터와 동일한 방식으로 동작하기 때문에,
위에서 언급한 공유 포인터의 순환 참조의 문제에서 자유롭지 않습니다.
그래서 언리얼 C++은 언리얼 오브젝트를 약하게 참조하는
TWeakObjectPtr이라는 별도의 라이브러리를 제공하고 있습니다.
특정 언리얼 오브젝트를 참조할 때,
반드시 소유권이 필요하지 않는 경우에는 약 참조를 걸어주는 것을 추천합니다.
예를 들어 UI의 리스트박스에서 언리얼 오브젝트의 목록을 보여주고 싶을 때
TWeakObjectPtr을 사용해 언리얼 오브젝트를 약참조(Weak Referencing)하는 것이 바람직합니다.
일반 참조를 걸게되면 UI가 띄워져 있는 동안에는 UI에서 보여지는 모든 언리얼 오브젝트의 레퍼런스 카운팅이 올라가게되므로, 언리얼 오브젝트를 삭제해도 GC시스템에서 회수가 일어나지 않습니다.
'Unreal > Study' 카테고리의 다른 글
언리얼 엔진 이해하기 (0) | 2019.06.07 |
---|---|
유니티 - 언리얼 차이점 (0) | 2019.06.07 |
11 직렬화=시리얼라이제이션 + 로직 +FArchive (0) | 2019.05.31 |
10. 언리얼 C++ 딜리게이트 + 종류 + 바인딩 (0) | 2019.05.30 |
9. INI 설정과 런타임 애셋 로딩 + INI구성 + 레퍼런싱 + config (0) | 2019.05.29 |