UE4BeginPlay执行流程

文章目录

  • UE4 BeginPlay执行流程
    • 源代码
      • UGameInstance::StartPlayInEditorGameInstance:
      • UWorld::BeginPlay
      • AGameModeBase::StartPlay
      • AGameStateBase::HandleBeginPlay
      • AWorldSettings::NotifyBeginPlay
      • AActor::DispatchBeginPlay

UE4 BeginPlay执行流程



Created with Raphaël 2.3.0 开始 UGameInstance::StartPlayInEditorGameInstance() 调用UWorld::BeginPlay() UWorld::BeginPlay()调用AGameModeBase::StartPlay() AGameModeBase::StartPlay()调用AGameStateBase::HandleBeginPlay() AGameStateBase::HandleBeginPlay()调用AWorldSettings::NotifyBeginPlay() AWorldSettings::NotifyBeginPlay()遍历所有Actor并调用AActor::DispatchBeginPlay(bool bFromLevelStreaming)) AActor::DispatchBeginPlay(bool bFromLevelStreaming)) 调用AActor::BeginPlay() 结束
Actor BeginPlay调用流程

源代码

UGameInstance::StartPlayInEditorGameInstance:

FGameInstancePIEResult UGameInstance::StartPlayInEditorGameInstance(ULocalPlayer* LocalPlayer, const FGameInstancePIEParameters& Params)
{if (!Params.EditorPlaySettings){return FGameInstancePIEResult::Failure(NSLOCTEXT("UnrealEd", "Error_InvalidEditorPlaySettings", "Invalid Editor Play Settings!"));}if (PIEStartTime == 0){PIEStartTime = Params.PIEStartTime;}BroadcastOnStart();UEditorEngine* const EditorEngine = CastChecked<UEditorEngine>(GetEngine());// for clients, just connect to the serverif (Params.NetMode == PIE_Client){FString Error;FURL BaseURL = WorldContext->LastURL;FString URLString(TEXT("127.0.0.1"));uint16 ServerPort = 0;if (Params.EditorPlaySettings->GetServerPort(ServerPort)){URLString += FString::Printf(TEXT(":%hu"), ServerPort);}if (Params.EditorPlaySettings->IsNetworkEmulationEnabled()){if (Params.EditorPlaySettings->NetworkEmulationSettings.IsEmulationEnabledForTarget(NetworkEmulationTarget::Client)){URLString += Params.EditorPlaySettings->NetworkEmulationSettings.BuildPacketSettingsForURL();}}if (EditorEngine->Browse(*WorldContext, FURL(&BaseURL, *URLString, (ETravelType)TRAVEL_Absolute), Error) == EBrowseReturnVal::Pending){EditorEngine->TransitionType = ETransitionType::WaitingToConnect;}else{return FGameInstancePIEResult::Failure(FText::Format(NSLOCTEXT("UnrealEd", "Error_CouldntLaunchPIEClient", "Couldn't Launch PIE Client: {0}"), FText::FromString(Error)));}}else{// we're going to be playing in the current world, get it ready for playUWorld* const PlayWorld = GetWorld();FString ExtraURLOptions;if (Params.EditorPlaySettings->IsNetworkEmulationEnabled()){NetworkEmulationTarget CurrentTarget = Params.NetMode == PIE_ListenServer ? NetworkEmulationTarget::Server : NetworkEmulationTarget::Client;if (Params.EditorPlaySettings->NetworkEmulationSettings.IsEmulationEnabledForTarget(CurrentTarget)){ExtraURLOptions += Params.EditorPlaySettings->NetworkEmulationSettings.BuildPacketSettingsForURL();}}// make a URLFURL URL;// If the user wants to start in spectator mode, do not use the custom play world for nowif (EditorEngine->UserEditedPlayWorldURL.Len() > 0 || Params.OverrideMapURL.Len() > 0){FString UserURL = EditorEngine->UserEditedPlayWorldURL.Len() > 0 ? EditorEngine->UserEditedPlayWorldURL : Params.OverrideMapURL;UserURL += ExtraURLOptions;// If the user edited the play world url. Verify that the map name is the same as the currently loaded map.URL = FURL(NULL, *UserURL, TRAVEL_Absolute);if (URL.Map != PIEMapName){// Ensure the URL map name is the same as the generated play world map name.URL.Map = PIEMapName;}}else{// The user did not edit the url, just build one from scratch.URL = FURL(NULL, *EditorEngine->BuildPlayWorldURL(*PIEMapName, Params.bStartInSpectatorMode, ExtraURLOptions), TRAVEL_Absolute);}// If a start location is specified, spawn a temporary PlayerStartPIE actor at the start location and use it as the portal.AActor* PlayerStart = NULL;if (!EditorEngine->SpawnPlayFromHereStart(PlayWorld, PlayerStart)){// failed to create "play from here" playerstartreturn FGameInstancePIEResult::Failure(NSLOCTEXT("UnrealEd", "Error_FailedCreatePlayFromHerePlayerStart", "Failed to create PlayerStart at desired starting location."));}if (!PlayWorld->SetGameMode(URL)){// Setting the game mode failed so bail return FGameInstancePIEResult::Failure(NSLOCTEXT("UnrealEd", "Error_FailedCreateEditorPreviewWorld", "Failed to create editor preview world."));}FGameInstancePIEResult PostCreateGameModeResult = PostCreateGameModeForPIE(Params, PlayWorld->GetAuthGameMode<AGameModeBase>());if (!PostCreateGameModeResult.IsSuccess()){return PostCreateGameModeResult;}// Make sure "always loaded" sub-levels are fully loadedPlayWorld->FlushLevelStreaming(EFlushLevelStreamingType::Visibility);PlayWorld->CreateAISystem();PlayWorld->InitializeActorsForPlay(URL);// calling it after InitializeActorsForPlay has been called to have all potential bounding boxed initializedFNavigationSystem::AddNavigationSystemToWorld(*PlayWorld, LocalPlayers.Num() > 0 ? FNavigationSystemRunMode::PIEMode : FNavigationSystemRunMode::SimulationMode);// @todo, just use WorldContext.GamePlayer[0]?if (LocalPlayer){FString Error;if (!LocalPlayer->SpawnPlayActor(URL.ToString(1), Error, PlayWorld)){return FGameInstancePIEResult::Failure(FText::Format(NSLOCTEXT("UnrealEd", "Error_CouldntSpawnPlayer", "Couldn't spawn player: {0}"), FText::FromString(Error)));}}UGameViewportClient* const GameViewport = GetGameViewportClient();if (GameViewport != NULL && GameViewport->Viewport != NULL){// Stream any levels now that need to be loaded before the game startsGEngine->BlockTillLevelStreamingCompleted(PlayWorld);}if (Params.NetMode == PIE_ListenServer){// Add portuint16 ServerPort = 0;if (Params.EditorPlaySettings->GetServerPort(ServerPort)){URL.Port = ServerPort;}// start listen server with the built URLPlayWorld->Listen(URL);}PlayWorld->BeginPlay();}// Give the deprecated method a chance to fail as wellFGameInstancePIEResult StartResult = FGameInstancePIEResult::Success();if (StartResult.IsSuccess()){PRAGMA_DISABLE_DEPRECATION_WARNINGSStartResult = StartPIEGameInstance(LocalPlayer, Params.bSimulateInEditor, Params.bAnyBlueprintErrors, Params.bStartInSpectatorMode) ?FGameInstancePIEResult::Success() :FGameInstancePIEResult::Failure(NSLOCTEXT("UnrealEd", "Error_CouldntInitInstance", "The game instance failed to Play/Simulate In Editor"));PRAGMA_ENABLE_DEPRECATION_WARNINGS}return StartResult;
}

UWorld::BeginPlay

void UWorld::BeginPlay()
{AGameModeBase* const GameMode = GetAuthGameMode();if (GameMode){GameMode->StartPlay();if (GetAISystem()){GetAISystem()->StartPlay();}}OnWorldBeginPlay.Broadcast();#if WITH_CHAOSif(PhysicsScene){PhysicsScene->OnWorldBeginPlay();}
#endif
}

AGameModeBase::StartPlay

void AGameModeBase::StartPlay()
{GameState->HandleBeginPlay();
}

AGameStateBase::HandleBeginPlay

void AGameStateBase::HandleBeginPlay()
{bReplicatedHasBegunPlay = true;GetWorldSettings()->NotifyBeginPlay();GetWorldSettings()->NotifyMatchStarted();
}

AWorldSettings::NotifyBeginPlay

void AWorldSettings::NotifyBeginPlay()
{UWorld* World = GetWorld();if (!World->bBegunPlay){for (FActorIterator It(World); It; ++It){SCOPE_CYCLE_COUNTER(STAT_ActorBeginPlay);const bool bFromLevelLoad = true;It->DispatchBeginPlay(bFromLevelLoad);}World->bBegunPlay = true;}
}

FActorIterator 是actor的遍历器,它继承自 TActorIteratorBase<FActorIterator>TActorIteratorBase里面存储了World的指针,并重写了++符号,所以这样写能遍历Actor
ActorIteratorBase重写的方法:

void operator++()
{// Use local version to avoid LHSs as compiler is not required to write out member variables to memory.AActor*           LocalCurrentActor      = nullptr;int32             LocalIndex             = State->Index;TArray<UObject*>& LocalObjectArray       = State->ObjectArray;TArray<AActor*>&  LocalSpawnedActorArray = State->SpawnedActorArray;UWorld*           LocalCurrentWorld      = State->CurrentWorld;while(++LocalIndex < (LocalObjectArray.Num() + LocalSpawnedActorArray.Num())){if (LocalIndex < LocalObjectArray.Num()){LocalCurrentActor = static_cast<AActor*>(LocalObjectArray[LocalIndex]);}else{LocalCurrentActor = LocalSpawnedActorArray[LocalIndex - LocalObjectArray.Num()];}State->ConsideredCount++;ULevel* ActorLevel = LocalCurrentActor ? LocalCurrentActor->GetLevel() : nullptr;if ( ActorLevel&& static_cast<const Derived*>(this)->IsActorSuitable(LocalCurrentActor)&& static_cast<const Derived*>(this)->CanIterateLevel(ActorLevel)&& ActorLevel->GetWorld() == LocalCurrentWorld){// ignore non-persistent world settingsif (ActorLevel == LocalCurrentWorld->PersistentLevel || !LocalCurrentActor->IsA(AWorldSettings::StaticClass())){State->CurrentActor = LocalCurrentActor;State->Index = LocalIndex;return;}}}State->CurrentActor = NULL;State->ReachedEnd = true;
}

AActor::DispatchBeginPlay

void AActor::DispatchBeginPlay(bool bFromLevelStreaming)
{UWorld* World = (!HasActorBegunPlay() && !IsPendingKill() ? GetWorld() : nullptr);if (World){ensureMsgf(ActorHasBegunPlay == EActorBeginPlayState::HasNotBegunPlay, TEXT("BeginPlay was called on actor %s which was in state %d"), *GetPathName(), (int32)ActorHasBegunPlay);const uint32 CurrentCallDepth = BeginPlayCallDepth++;bActorBeginningPlayFromLevelStreaming = bFromLevelStreaming;ActorHasBegunPlay = EActorBeginPlayState::BeginningPlay;BeginPlay();ensure(BeginPlayCallDepth - 1 == CurrentCallDepth);BeginPlayCallDepth = CurrentCallDepth;if (bActorWantsDestroyDuringBeginPlay){// Pass true for bNetForce as either it doesn't matter or it was true the first time to even // get to the point we set bActorWantsDestroyDuringBeginPlay to trueWorld->DestroyActor(this, true); }if (!IsPendingKill()){// Initialize overlap stateUpdateInitialOverlaps(bFromLevelStreaming);}bActorBeginningPlayFromLevelStreaming = false;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/9791.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

算法第63天 单调栈3

84 柱状图中最大的矩形 给定 n 个非负整数&#xff0c;用来表示柱状图中各个柱子的高度。每个柱子彼此相邻&#xff0c;且宽度为 1 。 求在该柱状图中&#xff0c;能够勾勒出来的矩形的最大面积。 def largestRectangleArea(heights):res0for i in range(len(heights)):left…

【Linux】如何查看Linux命令的使用方法

man cd | col -b > cdhelp.txt #查询帮助文档 man 命令用于查看 Linux 系统的在线帮助文档&#xff08;man page&#xff09;。man 是 “manual” 的缩写&#xff0c;通过它&#xff0c;你可以访问关于各种命令、函数、配置文件等的详细文档。 cd 是一个常用的命令&#xf…

YOLO数据集制作(一)|Labelme标注的矩形框json文件转txt

以下教程用于将使用Labelme软件标注生成的json格式文件转成YOLO使用的txt格式&#xff0c;适用场景&#xff1a;矩形框 使用方法&#xff1a;将json文件夹路径填到json_dir后&#xff0c; 将保存转化后txt的路径填入txt_dir后&#xff0c;运行即可转化。 运行完成后会生成label…

io流、字节流字符流区别

想象你正在做一道菜&#xff0c;需要准备材料和烹饪步骤。IO流就像是你在做菜时使用的工具&#xff0c;而字节流和字符流则是不同类型的工具&#xff0c;适合不同的操作。 首先&#xff0c;字节流就像是你使用的普通厨具&#xff0c;比如刀、砧板等。它们可以处理任何类型的食…

Linux-解压缩文件命令(gzip、zip、unzip、tar、jar)

目录 1、压缩解压命令 1.1 gzip、gzcat、gunzip 1.2 bzip2、bzcat、bunzip2 1.3 zip和unzip 2、tar命令 3、jar命令 1、压缩解压命令 1.1 gzip、gzcat、gunzip gzip 命令 压缩文件后缀.gz gzcat 用来显示压缩的文本文件内容 gunzip 用来解压文件 把/home目录下的t…

使用向量数据库 Zilliz Cloud 时常用到的术语和概念

AUTOINDEX 为满足用户不同需求,Zilliz Cloud 提供 2 种类型的集群 CU:性能型和容量型。但是,为不同类型 CU 集群中的 Collection 创建索引时,通常需要根据所选择的 CU 类型调整索引参数。为了方便创建索引,免去调节参数的麻烦,Zilliz Cloud 使用 AUTOINDEX 的索引类型。…

OpenAI 正在开发一种可以防止版权诉讼的工具

OpenAI 正在开发一种名为 Media Manager 的工具&#xff0c;该工具将使内容创建者和所有者能够确定他们是否愿意将自己的内容用于 ML 研究和 AI 模型训练。 Media Manager 将做什么&#xff1f; 明年推出的 Media Manager 将使内容创作者和所有者能够更好地控制他们的工作是否…

Oracle体系结构初探:闪回技术

在Oracle体系结构初探这个专栏中&#xff0c;已经写过了REDO、UNDO等内容。觉得可以开始写下有关备份恢复的内容。闪回技术 — Oracle数据库备份恢复机制的一种。它可以在一定条件下&#xff0c;高效快速的恢复因为逻辑错误&#xff08;误删误更新等&#xff09;导致的数据丢失…

岩点×数说故事×小红书 | 发布《中国攀岩行业分析报告》

从下班健身到下班攀岩&#xff0c;从“鸡娃”到岩馆“溜娃”&#xff0c;被奥运“正名”的攀岩运动&#xff0c;在国内熬过了萌芽阶段&#xff0c;悄然开出了花。2023年&#xff0c;各类重磅攀岩赛事重启、线下岩馆疯狂扩张&#xff0c;小众攀岩正式进入大众视野&#xff0c;风…

249 基于matlab的MED、OMEDA、MOMEDA、MCKD信号处理方法

基于matlab的MED、OMEDA、MOMEDA、MCKD信号处理方法。最小熵反褶积(MED)&#xff0c;最优最小熵反卷积调整卷积 (OMEDA),多点最优最小熵解卷积调整&#xff08;Multipoint Optimal Minimum Entropy Deconvolution Adjusted&#xff0c;MOMEDA&#xff09;&#xff0c;最大相关峭…

Vue2 组件通信方式

props/emit props 作用&#xff1a;父组件通过 props 向子组件传递数据parent.vue <template><div><Son :msg"msg" :pfn"pFn"></Son></div> </template><script> import Son from ./son export default {name: …

winform图书管理系统

winform图书管理系统说明文档 运行前附加数据库.mdf&#xff08;或sql生成数据库&#xff09; 主要技术&#xff1a; 图书管理员 读者管理 图书管理 添加 修改 删除 查看 入库 书册列表 书册管理用户管理退出 借书 还书 系统管理员 修改图书管理权限 项目获取&#xff1a;…

LeetCode 209 长度最小的子数组(滑动窗口and暴力)

、 法一&#xff1a;滑动窗口 //使用滑动窗口来解决问题 //滑动窗口的核心点有&#xff1a; /*1.窗口内是什么&#xff1f;2.如何移动窗口的起始位置&#xff1f;3.如何移动窗口的结束位置&#xff1f;4.两个指针&#xff0c;怎么判断哪个指针是终止指针&#xff0c;哪个指针…

代码随想录算法训练营第六十天| LeetCode647. 回文子串 、516.最长回文子序列

一、LeetCode647. 回文子串 题目链接/文章讲解/视频讲解&#xff1a;https://programmercarl.com/0647.%E5%9B%9E%E6%96%87%E5%AD%90%E4%B8%B2.html 状态&#xff1a;已解决 1.思路 这道题我只想出来了暴力解法&#xff0c;动规解法并没有想出来。根据视频讲解才把它想出来。…

针对macOS上的maven安装配置

这篇博客将向读者介绍如何安装和配置Maven。Maven是一个强大的项目管理工具&#xff0c;广泛用于Java项目的构建、依赖管理和项目报告生成。它可以极大地简化项目的构建过程&#xff0c;并帮助开发人员管理项目的各种依赖项。 什么是Maven&#xff1f; Maven是一个基于项目对…

C++——命名空间

c ——命名空间 前言一.命名空间命名空间的进一步拓展 二.io流特性 前言 ** 好久不见&#xff0c;甚是想念~今天我们讲解的是关于c命名空间的一些知识点&#xff0c;这只是开胃小菜哦&#xff0c;期待我们后面更深入知识的灵魂碰撞吧 ** 一.命名空间 怎么形容呢~命名空间出现…

Rocketmq如何保证消息不丢失和幂等性

生产者 生产者通过RocketMQ提供的事务消息&#xff08;两阶段提交&#xff09;能保证消息的一致性。 第一阶段给Broker发送一个半事务消息&#xff0c;半事务消息是不能消费的消息&#xff0c;broker已经收到生产者发送的消息&#xff0c;但是并未收到生产者的二次确认&#x…

SpringBoot异步接口实现:提高系统的吞吐量

前言 Servlet 3.0之前&#xff1a;每一次Http请求都由一个线程从头到尾处理。 Servlet 3.0之后&#xff0c;提供了异步处理请求&#xff1a;可以先释放容器分配给请求的线程与相关资源&#xff0c;减轻系统负担&#xff0c;从而增加服务的吞吐量。 在springboot应用中&#…

JavaScript使用 BigInt

在 JavaScript 中&#xff0c;最大的安全整数是 2 的 53 次方减 1&#xff0c;即 Number.MAX_SAFE_INTEGER&#xff0c;其值为 9007199254740991。这是因为 JavaScript 中使用双精度浮点数表示数字&#xff0c;双精度浮点数的符号位占 1 位&#xff0c;指数位占 11 位&#xff…

2024C题生物质和煤共热解问题的研究 详细思路

背景 随着全球能源需求的不断增长和对可再生能源的追求&#xff0c;生物质和煤共热解作为一种潜在的能源转化技术备受关注。生物质是指可再生能源&#xff0c;源自植物和动物的有机物质&#xff0c;而煤则是一种化石燃料。** 在共热解过程中&#xff0c;生物质和煤在高温和缺氧…