找回密码
 立即注册
查看: 593|回复: 0

UE4/5 开发面试题:Unreal

[复制链接]
发表于 2023-5-21 14:11 | 显示全部楼层 |阅读模式
1.游戏模式和游戏实例


  • GameInstance:游戏实例,架构上超出于最顶端,实例里的数据不会随着关卡的切换而消掉。感化是为游戏保留临时的全局数据,或者自定义需要在全局使用的逻辑。在“项目设置 - 项目 - 地图和模式 - 游戏实例 - 游戏实例类”处改削。游戏运行时蓝图使用“GetGameInstance”节点获取,C++ 使用“UGameplayStatics::GetGameInstance()”函数获取。
  • GameMode:设置游戏法则,在当前地图内生效。它可以规定玩家和不雅观众数量,以及制定玩家进入游戏的方式,包含 Spawn 地址和生成/重生等行为的法则。联网状态下,只存在于处事端,客户端不能访谒。在“项目设置 - 项目 - 地图和模式 - 默认模式 - 默认游戏模式”或“世界场景设置 - 游戏模式 - 游戏模式重载”或“Config - DefaultEngine.ini - GameMapsSettings - GlobalDefaultGameMode”处均可改削。
  • DefaultPawn:角色的外在表示类,是玩家在游戏中的物理代表,可包含自身的移动法则和其他游戏逻辑,处事端和客户端都存在一份,同时保持同步。在 GameMode 中改削。
  • HUD:是显示屏幕上覆盖元素的基本对象。游戏中每个由人类控制的玩家都有本身的AHUD类实例,这个实例会绘制到个人视口上。只存在于客户端。在 GameMode 中改削。
  • PlayerController:非常重要的一个类,拥有 Pawn 并设置其行为法则,处事器上拥有所有玩家的PlayerController,而当地客户端则只有当前玩家的 PlayerController 。关联了客户端和处事端,通过该类,客户端可以向处事端发送请求。在 GameMode 中改削。
  • GameState:数据的全局打点,处事端和客户端都存在一份,它包含要复制到游戏中的每个客户端的信息,凡是包含游戏分数、比赛是否已开始和基于世界场景玩家人数要生成的AI数量等的信息,以及其他特定于游戏的信息。凡是用来保持数据的同步,也可将此中的部门数据设置为分歧步。在 GameMode 中改削。
  • PlayerState:角色数据,该类需要通过 PlayerController 来访谒。PlayerState 中保留当前玩家的一些信息,例如玩家姓名或得分、当前等级及生命值等,是对应玩家的数据容器。对于多人游戏,所有玩家的 PlayerState存在于所有机器上(与 PlayerController 分歧),而且可以将数据从处事器复制到客户端以保持同步。在GameMode 中改削。
客户端处事器
GameMode——存在
DefaultPawn存在存在
HUD存在不存在
PlayerController只有当前玩家拥有所有玩家
GameState存在存在
PlayerState拥有所有玩家拥有所有玩家
2.动态材质实例的感化[1]

动态材质实例 (MID) 是可以在游戏期间(在运行时)进行计算的实例化材质。这暗示在游戏期间,您可使用脚本(颠末编译的代码或蓝图可视脚本)来更改材质的参数,从而在游戏中改变该材质。这种材质的可能应用场所数不胜数,既可显示分歧程度的损坏,也可更改绘图作业以混入分歧皮肤纹理来回应面部表情。
3.单播委托、多播委托和动态委托的区别?[2]

委托是一种泛型但类型安全的方式,可在 C++ 对象上调用成员函数。可使用委托动态绑定到任意对象的成员函数,之后在该对象上调用函数,即使调用法式不知对象类型也可进行操作。

  • 单播委托:只能绑定一个委托函数,绑定的委托函数可以有返回值,可接受分歧数量的参数(最多撑持9个函数参数),委托实例必需绑定在其声明时所定义的同返回类型和参数列表的函数,静态委托执行前最好查抄是否绑定,否则会导致法式崩溃,如果反复进行绑定,会覆盖上一次的绑定。
  • 多播委托:拥有大部门与单播委托不异的功能。它们只拥有对对象的弱引用,可以与布局体一起使用,可以四处轻松复制等等。多播委托可以绑定多个委托函数,可以长途加载/保留和触发,但多播委托函数不能使用返回值。它们最适合用来四处轻松传递一组委托。多播委托在广播执行时不必然是按照绑定挨次来的,在广播执行时,不需要判断是否绑定了委托函数,直接广播执行即可。
  • 动态委托:动态委托包含动态单播和动态多播,撑持蓝图序列化,即可以在蓝图中使用,其函数可按定名查找,但其执行速度比常规委托慢。
委托可绑定的委托函数参数返回值序列化
单播一个撑持撑持不撑持
多播多个撑持不撑持不撑持
动态单播一个撑持撑持撑持
动态多播多个撑持不撑持撑持
//单播委托
DECLARE_DELEGATE(FDelegate);                                             //声明
FDelegate Delegate;                                                      //定义
Delegate.ExecuteIfBound();                                               //调用
ActorReference->Delegate.BindUObject(this, &AMyActor::DelegateFunction); //绑定
//多播委托
DECLARE_MULTICAST_DELEGATE(FMulticastDelegate);
FMulticastDelegate MulticastDelegate;
MulticastDelegate.Broadcast();
ActorReference->MulticastDelegate.AddUObject(this, &AMyActor::MulticastDelegateFunction);
//动态单播委托
DECLARE_DYNAMIC_DELEGATE(FDynamicDelegate);
FDynamicDelegate DynamicDelegate;
DynamicDelegate.ExecuteIfBound();
ActorReference->DynamicDelegate.BindDynamic(this, &AMyActor::DynamicDelegateFunction);
//动态多播委托
DECLARE_DYNAMIC_MULTICAST_DELEGATE(FDynamicMulticastDelegate);
FDynamicMulticastDelegate DynamicMulticastDelegate;
DynamicMulticastDelegate.Broadcast();
ActorReference->DynamicMulticastDelegate.AddDynamic(this, &AMyActor::DynamicMulticastDelegateFunction);
4.委托的底层道理

委托是一种不雅察看者模式,也被称为代办代理,用于降低分歧对象之间的耦合度,两个有关联的对象不合错误彼此的行为进行监听,而是通过委托来间接的成立联系,监听者将需要响应的函数绑定到委托对象上,使得委托在触发时调用所绑定的函数。
在 UE 中,委托机制的道理斗劲简单,就是在委托类的内部保留了函数指针,需要执行这些委托的时候就传入所需的参数给保留的函数指针,从而调用绑定的函数。但实现上稍显复杂,因为要解决两个问题:[3]

  • 需要撑持具有任意类型以及数量不限的参数列表的函数
  • 需要撑持多种类型函数,如 lambda 匿名函数、C++ 原始成员函数、基于共享指针的成员函数、原始全局函数(包罗静态成员函数)、基于 UFunction 的成员函数、基于 UObject 的成员函数
为了撑持任意类型使用了 template 模板,撑持数量不限的参数则使用了可变参数以及 Tuple 元组。而对于撑持多种类型函数,只是尽可能地复用代码的基础上为这些分歧的类型函数编写分歧的绑定、调用方式。UE 实现上有三个核心的类:模板类 TDelegate、原生类 FDelegateBase 和接口类 IDelegateInstance,TDelegate 提供给用户使用的顶层接口,FDelegateBase 提供分歧类型函数实现时通用的底层接口,IDelegateInstance 的各种派生类中完成 TDelegate 中各种顶层接口的实现。
5.如何保持新建的UObject对象不被自动GC垃圾回收?[4]

(1)在普通的C++类中新建UObject对象后,使用AddToRoot()函数可以庇护对象不被自动回收,移除庇护时使用RemoveFromRoot()并把对象指针置为nullptr即可由引擎自动回收;
UMyObject* MyObject=NewObject<UMyObject>();
MyObject->AddToRoot();                     //庇护对象不被回收
MyObject->RemoveFromRoot();
MyObject=nullptr;                          //交给引擎回收对象
(2)如果是在担任自UObject类中新建UObject对象后,使用UPROPERTY宏标识表记标帜一下对象指针变量也可以庇护对象不被自动回收,在该类被销毁时,新建的对象也会被引擎自动回收;
UCLASS()
class UMyObject : public UObject{
    GENERATED_BODY()
    UPROPERTY()
    class UItemObject* ItemObject;
}
(3)使用FStreamableManager加载资源时,将bManageActiveHandle设置为true也可以防止对象被回收;
FSoftObjectPath AssetPaths(TEXT(”[资源路径]”));
FStreamableManager& AssetLoader = UAssetManager::GetStreamableManager();
TSharedPtr<FStreamableHandle> Handle = AssetLoader.RequestSyncLoad(AssetPath, true);//加载资源到内存中,bManageActiveHandle=true
UObject* Obj = Handle->GetLoadedAsset();
Handle->ReleaseHandle();//从内存中释放资源
(4)FGCObjectScopeGuard在指定代码区域内保持对象;
{
    FGCObjectScopeGuard(UObject* GladOS = NewObject<...>(...));
    GladOS->SpawnCell();
    RunGC();
    GladOS->IsStillAlive();
}
6. 三种智能指针


  • 共享指针(TSharedPtr)允许多个该类型的指针指向同一块内存,采用引用计数器的方式,统计所有指向同一块内存的指针变量的数量,当新的指针变量生命初始化并指向同一块内存,拷贝函数拷贝和赋值操作时引用计数器会自增加,当指针变量生命周期结束调用析构函数时,引用计数器会自减少。引用计数器减少至0时,释放指向的内存。共享引用(TShareRef)和共享指针的区别是共享指针可以为 NULL,而共享引用不能为 NULL 。
  • 弱指针(TWeakPtr`TSharedPtr`)主要是为了配合共享指针而引入的一种智能指针,TWeakPtr 没有指针的行为,没有重载间接引用操作符(->)和解除引用操作符(*),它可以通过 TSharedPtr 和 TSharedRef 来初始化,但只引用,不计数,不拥有内存的所有权,不会对 TSharedPtr 和 TSharedRef 的共享引用计数器发生影响,也不影响其生命周期,但会在控制块的 WeakReferenceCount 属性中统计弱指针引用数量。
  • 独一指针(TUniquePtr)仅会显式拥有其引用的对象。仅有一个独一指针指向给定资源,因此独一指针可转移所有权,但无法共享。复制独一指针的任何测验考试都将导致编译错误。独一指针超出范围时,其将自动删除其所引用的对象。
TSharedPtr<Person> sp = MakeShared<Person>();     //创建共享指针
TSharedRef<Person> sr = sp.ToSharedRef();         //创建共享引用
TWeakPtr<Person> wp = sp;                         //创建弱指针
int32 use_count = sp.GetSharedReferenceCount();   //共享指针计数
TUniquePtr<Person> up = MakeUnique<Person>();     //创建独一指针
7.智能指针的循环引用

在使用基于引用计数的 TSharedPt r智能指针时,为了防止循环引用带来的内存泄漏问题,可以让引用链上的一方持用弱智能指针 TWeakPtr 。弱智能指针不会影响共享引用计数器。
8. 如何使用ParallelFor提高速度

ParallelFor允许我们在一分钟内对任何 for 循环进行多线程措置,从而通过在多个线程之间拆分工作来划分执行时间。
//例1
ParallelFor(num, [&](int32 i) {sum += i; });
//例2
FCriticalSection Mutex;
ParallelFor(Input.Num(), [&](int32 Idx){
    if(Input[Idx] % 5 == 0){
        Mutex.Lock();
        Output.Add(Input[Idx]);
        Mutex.Unlock();
    }
});
9.TMap的实现道理

TMap是用基于数组的哈希表实现的,查询效率高,添加、删除效率低,查询的时间复杂度是 O(1) 。TMap的排序采用的快速排序 , 时间复杂度为 O(nlog_{2}n) 。
数据布局查询时间复杂度长处错误谬误
map红黑树O(logn)内部自动排序,查询、添加、删除效率不异空间占用较大
unordered_map哈希表O(1)查询效率高内部元素无序杂乱添加、删除效率低
TMap哈希表O(1)查询效率高内部元素无序杂乱添加、删除效率低
10.法线是存储在什么空间[5]

切线空间。对法线做空间变换时,若模型本身做了 XYZ 轴上的非均匀缩放,则法线值会发生偏移,但切线并不会受到影响,所以相较于模型空间,一般常用切线空间存储法线数据。切线空间存储的是相对法线信息,在切线空间中,每个法线标的目的地址的坐标空间是纷歧样的,便是概况每点各自的切线空间,这种法线纹理其实存储了每个点在各自的切线空间中的 shading 法线偏移(扰动)标的目的,如果一个点的法线标的目的不变,那么在其切线空间中,新的法线标的目的就是 Z 轴标的目的。即值为(0,0,1),映射到颜色即(0.5,0.5,1)浅蓝色。


11.虚幻中有哪几种主要线程[6]

游戏线程(GameThread):承载游戏逻辑、运行流程的工作,也是其它线程的数据倡议者。在FEngineLoop::Tick 函数执行每帧逻辑的更新。在引擎启动时会把 GameThread 的线程 id 存储到全局变量GGameThreadId 中,且稍后会设置到 TaskGraph 系统中。
衬着线程(RenderThread):RenderThread 在 TaskGraph 系统中有一个任务队列,其他线程(主要是GameThread)通过宏 ENQUEUE_RENDER_COMMAND 向该队列中填充任务,RenderThread 则不竭从这个队列中取出任务来执行,从而生成与平台无关的 Command List(衬着指令列表)。
RHI线程(Render Hardware Interface Thread):RenderThread 作为前端(frontend)发生的 Command List 是平台无关的,是抽象的图形 API 调用;而 RHIThread 作为后端(backend)会执行和转换衬着线程的Command List 成为指定图形 API 的调用(称为 Graphical Command),并提交到 GPU 执行。RHI 线程的工作是转换衬着指令到指定图形 API,创建、上传衬着资源到 GPU。
12.游戏线程和衬着线程的同步[7]

当 GameThread 与 RenderThread 同步时,GameThread 会创建一个 FNullGraphTask 空任务,放到RenderThread 的 TaskGraph 队列中让其执行,在 FRenderCommandFence 的 Wait 函数中,会查抄投递给RenderThread 的 CompletionEvent 是否被执行,如果没有执行则调用 GameThreadWaitForTask 函数来阻塞等待。
13.CharacterMovementComponent 如何更改移动速度

CharacterMovementComponent 的工作原则是使用加速度驱动速度,通过速度表示具体 Actor 的移动,输入当前加速度的标的目的和最大加速度的百分比来更改移动速度。
14.多线程 Task Graph[8]

TaskGraph 是 UE 中基于任务的并发机制。可以创建任务在指定类型的线程中执行,同时提供了等待机制,其强大之处在于可以调剂一系列有依赖关系的任务,这些任务组成了一个有向无环的任务网络(DAG),而且任务的执行可以分布在分歧的线程中。
void ATestTaskGraphActor::CreateTask(FString TaskName, const TArray<TGraphTask<FWorkTask>*>& Prerequisites,
  const TArray<TGraphTask<FWorkTask>*>& ChildTasks){    //FWorkTask为自定义的类
    FGraphEventArray PrerequisiteEvents;
    TArray<TGraphTask<FWorkTask>*> ChildEvents;
    for (auto Item : Prerequisites)
        PrerequisiteEvents.Add(Item->GetCompletionEvent());
    for (auto Item : ChildTasks)
        ChildEvents.Add(Item);
    TGraphTask<FWorkTask>::CreateTask(&PrerequisiteEvents).ConstructAndDispatchWhenReady(TaskName, ChildEvents, this);
}
15.后措置之 bloom

泛光(Bloom)是一种现实世界中的光现象,通过它能够以较为适度的衬着性能成本极大地增加衬着图像的真实感。用肉眼不雅察看暗中布景下非常敞亮 的物体时会看到泛光效果。泛光可以用一个高斯模糊来实现。为了提高质量,我们将多个分歧半径的高斯模糊组合起来。为了获得更好的性能,我们在大大降低的分辩率下进行很宽范围的模糊。通过改变模糊效果的组合方式,我们可以进行更多的控制,取得更高的质量。为了获得最佳的性能,应该使用高分辩率模糊(小值)来实现较窄的模糊,而主要使用低分辩率模糊 (大值)实现较宽的模糊。
16.后措置之轮廓描边

方式(1):对需要描边的物体开启自定义深度缓存,物体地址区域会呈现填充的具有深度信息的缓存区,通过后期措置对相邻像素进行采样来执行简单的深度斗劲,如果邻居有深度信息,但像素没有,就将其着色为轮廓线颜色。


方式(2):获取场景法线向量后,通过此中一个做一点点UV偏移,是两个成果做差,颜色值越接近,插值越小,相反越大,而一般需要描边的位置就是向量相差较大的像素点,再用基础颜色加上这个差值就会呈现描边效果。[9]


17.蓝图大量连线为何会比 C++ 慢很多[10]

蓝图的消耗主要是在节点之间,蓝图连线触发的消耗是一致的,但节点运行的消耗是通过C++ , 节点分歧就有所分歧 ,所以蓝图中连线很多时会显著降低运行效率。
18.模型闪烁问题如何解决

当两个面共面时,会呈现闪面现象。使用UE4材质中 Pixel Depth Offset 节点,进行像素偏移,达到共面不闪面的效果。
19.虚幻内使用的光照模型

PBR,基于物理的光照。
20.slate中常用的控件

SHorizontalBox:程度框
SVerticalBox:垂直框
SUniformGridPanel:统一网格面板,均匀地垂直和程度分发子控件的面板
SWrapBox:包抄盒,程度摆列控件的盒
21.反射的感化

反射实现来撑持引擎的动态功能,如垃圾回收、序列化、网络复制和蓝图/C++通信等。
22.布局体中是否可以使用 UFUNCTION()

不成以。反射系统不撑持布局体中的函数,即使在 C++ 类和布局中实际上也具有与类不异的功能,UE4约定将布局限制为仅包含数据布局,这种做法实际上赋予布局更多的存在感。
23. UE 中垃圾回收的道理

UE4 采用了标识表记标帜-清扫的垃圾回收方式,是一种经典的垃圾回收方式。一次垃圾回收分为两个阶段。第一阶段从一个根调集出发,遍历所有可达对象,遍历完成后就能标识表记标帜出可达对象和不成达对象了,这个阶段会在一帧内完成。第二阶段会渐进式的清理这些不成达对象,因为不成达的对象将永远不能被访谒到,所以可以分帧清理它们,避免一下子清理很多 UObject,比如 map 卸载时,发生明显的卡顿。
24. UE 中有哪几种容器

TArray:是 UE4 中最常用的容器类,负责同类型其他对象(称为”元素”)序列的所有权和组织。由于TArray是一个序列,其元素的排序定义明确,其函数用于确定性地独霸此类对象及其挨次。
TMap:继TArray之后,UE4 中最常用的容器是TMap。TMap主要由两个类型定义(一个键类型和一个值类型),以关联对的形式存储在映射中。与TSet类似,它们的布局均基于对键进行散列运算。但与TSet分歧的是,此容器将数据存储为键值对(TPair<KeyType, ValueType>),只将键用于存储和获取。
TSet:是一种快速容器类,(凡是)用于在排序不重要的情况下存储独一元素。TSet类似于TMap和TMultiMap,但有一个重要区别:TSet是通过对元素求值的可覆盖函数,使用数据值本身作为键,而不是将数据值与独立的键相关联。TSet可以非常快速地添加、查找和删除元素(恒按时间)。默认情况下,TSet不撑持反复的键,但使用模板参数可激活此行为。
TList:简单的单链表模板。
25.gameplay 的框架[11]

UE 的游戏世界构成层级为:

  • GameInstance:游戏实例,由 GameEngine 缔造出来,主要用于打点世界切换,UI的加载,控制台命令和额外的逻辑,初始化/封锁引擎,改削GameMode,在线会话打点等一些全局性的内容。
  • World:游戏世界,常用布局体 FWorldContext 记录了游戏世界的各种信息使用在游戏世界切换等功能。
  • Level:一个游戏世界,可以分成多个 Level,比如将游戏场景分成一个 Level,灯光分成一个 Level 等等,这样有利于美术师进行场景搭建。关卡也分为 Persistence Level 和 Streaming Level,Persistence Level 是成立我们世界的主Level,Streaming Level 是作为部门内容的 Level 按照我们定义的法则加载到 Persistence Level 里。
  • GameMode:定义游戏法则,存在于每个 World/Level 中,而且只在处事器上,通过GameState来传递信息。
  • Actor:所有能放到游戏场景中的对象的基类都是 AActor。
  • Component:表示 Actor 的各种构成部门,UE 中的 Component 类能够附加到 Actor 上。
Gameplay 的 3C 概念,就是 Character、Camera、Control。Character 表示游戏世界中的玩家,拥有游戏世界中最复杂的行为,对于法式员来说,要负责措置角色的移动,动画,皮肤(装备)等。Camera 措置游戏视角,表示游戏世界,第一人称,第三人称,FOV,VFX,后措置,发抖等。Control 措置输入(来自鼠标键盘,手柄,模拟器等,以及输入模式例如单击、按住、双击),匹配输入逻辑(按键映射),措置回应输入的逻辑,UI交互,以及物理模拟,AI驱动等。
26.Unreal 中使用的光线追踪方式有哪些?
参考


  • ^https://www.bilibili.com/read/cv14351521
  • ^https://zhuanlan.zhihu.com/p/415867431
  • ^https://zhuanlan.zhihu.com/p/452566044
  • ^https://zhuanlan.zhihu.com/p/39564799
  • ^https://zhuanlan.zhihu.com/p/261667233
  • ^https://www.cnblogs.com/kekec/p/15464958.html
  • ^https://www.codeprj.com/blog/ebf9fe1.html
  • ^https://zhuanlan.zhihu.com/p/463272214
  • ^https://blog.csdn.net/qq_36696486/article/details/106319568
  • ^https://www.bilibili.com/read/cv4410697/
  • ^https://blog.csdn.net/qq_37856544/article/details/122187801

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

×
懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Unity开发者联盟 ( 粤ICP备20003399号 )

GMT+8, 2024-11-16 22:34 , Processed in 0.127837 second(s), 28 queries .

Powered by Discuz! X3.5 Licensed

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表