UE4联网游戏中让不同的客户端生成不同的Pawn类型
效果描述
一个服务器,两个客户端,让他们连接后分别生成不同的Pawn,并且在不同的位置生成。
意义
这是个项目需求,但是我发现如果能够彻底理解并制作出这个功能,会对虚幻4内置的网络功能以及一些重要的Gameplay 的类有更深入的了解。
目前已有解决方案
在Google上也搜了好久,但是相关信息并不多,比较靠谱的最终都指向同一个wiki页面:Spawn Different Pawns For Players in Multiplayer,但是他的解决方案有个缺陷:开发过程中不好测试——原因是他的方案需要一个外部存储的数据或者SaveGame,而开发时你不可能把一个工程复制两份。除非发布以后,把发布好的文件复制两份去测试,但这样也很麻烦。
我的解决方案概述
我采用了蓝图实现。区分不同的客户端主要是依靠OnPostLogin的连接顺序。
在GameMode中设置一个int变量作为用户索引,每次OnPostLogin后递增1。因为服务器(Listen Server)的PostLogin肯定是第一个(没哪个客户端的连接速度会快过本机),所以服务端连接时索引为0。而其他两个客户端在项目中并不需要立即区分其角色,只需要先分别给一个不同的角色,后面如果需要修改,在服务端提供界面进行手动修改即可。
预备知识
- 首先要熟悉UE4中的网络/复制/RPC相关概念以及引擎GameMode,PlayerController等类在网络环境中的表现,这些信息需要仔细阅读和理解官方的文档:
NetWorking and Multiplayer 其中的ActorReplication章节非常重要需要仔细阅读和理解,MultiPlayer in Blueprints章节是个引导概况性质的章节,也需要仔细读
- 其次要阅读GameModeBase源码,了解从PostLogin到真正生成Pawn之间都发生了什么,这里我已经总结成了一幅图:

具体实施步骤
准备工作
先使用ThirdPerson模板创建工程,然后创建ThirdPersonCharacter的三个子类,给不同的颜色作为标记,分别是红,绿,蓝,其中红色准备作为服务器的Pawn,其他两个作为客户端的。
以GameModeBase为基类创建一个GameMode蓝图,这里命名为MyGameMode
以PlayerController为基类窗机一个Controller蓝图,这里明明为MyPlayerController
创建一个Enum,命名为ClientType,包含三个值:Server, ClientA, ClientB
在ThirdPersionExampleMap中,删掉场景中的Character,然后把PlayerStart复制2个出来,摆放好位置,分别给三个PlayerStart的Player Start Tag属性设置为Server,ClientA,ClientB
关键步骤
MyPlayerController
在MyPlayerController中创建一个变量:
MyClientType:ClientType枚举类型,设置为Replicated,用于标记该PlayerController的类型
MyGameMode
在MyGameMode中,创建6个变量:
ClientIndex : int型,默认设置,用于标记不同的客户端连接,每次OnPostLogin后会递增1
PlayerStarts:PlayerStart引用类型,数组,其他默认,用于存放所有的PlayerStart
ServerPawnClass: Pawn类类型,默认设置,表示服务器端Pawn的类
ClientAPawnClass: Pawn类类型,默认设置,表示客户端APawn的类
ClientBPawnClass: Pawn类类型,默认设置,表示客户端BPawn的类
CurrentPlayer: MyPlayerController引用类型,默认设置,临时存放传入的PlayerController
创建两个关于获取PlayerStart的函数:
GetAllPlayerStarts

GetPlayerStartByTag:

这两个函数的含义如其名称,功能也比较简单。不过需要注意调用时机。有人可能会想,直接在BeginPlay里调用GetAllPlayerStarts就可以了,实际上这样不行,因为OnPostLogin事件会在BeginPlay之前发生。
右键搜索OnPostLogin, 创建Event OnPostLogin事件,连接如下图:

步骤释义:
- 当有玩家连接进来后(包括服务器自身连接自身),把PlayerController存入一个临时变量Current Player。
- 根据Client Index设置Current Player的MyClientType,依次设置为Server,ClientA,ClientB。
- 然后把Client Index自增1。
- 判断Controller是否已经拥有了Pawn,如果有则销毁。
- 调用Restart Player重新生成该Controller的Pawn(注意看上文中的流程图,Restart Player之后进行了什么操作)
到这步之后,Restart之后并没有改变要使用的Pawn的类。
根据上文中的流程图,Pawn的类是在GetDefaultPawnClassForController函数中获取的,在三处都使用了该函数来返回Pawn的类型,因此我们需要覆盖这个函数,点"Functions"中的Override按钮,覆盖该函数。
函数截图如下:

步骤释义:
获取PlayerController,转换为MyPlayerController,根据刚才存入的MyClientType来返回不同的Pawn类型。使用MyGameMode里的三个Pawn Class 变量。
到这里,Pawn类别已经可以正常区分了,但是起始点还不行,都是在同一个位置生成。下面要解决的就是区分PlayerStart。
看上文中的流程图,可以看到,在Restart Player函数中是通过调用Find Player Start函数来决定使用哪个PlayerStart。因此要覆盖FindPlayerStart函数。
在MyGameMode里的Functions里点"Override按钮,覆盖FindPlayerStart函数,覆盖后的截图如下:

因为之前已经给不同的PlayerController分配了不同的角色,所以这步比较简单,也是区别My Client Type,返回不同的PlayerStart即可。
到此为止就完成了
效果截图

UE4联网游戏中让不同的客户端生成不同的Pawn类型的更多相关文章
- Java游戏服务器成长之路——弱联网游戏篇(源码分析)
前言 前段时间由于公司的一款弱联网游戏急着上线,没能及时分享,现在基本做的差不多,剩下的就是测试阶段了(本来说元旦来分享一下服务器技术的).公司的这款游戏已经上线一年多了,在我来之前一直都是单机版本, ...
- UE4蓝图与C++交互——射击游戏中多武器系统的实现
回顾 学习UE4已有近2周的时间,跟着数天学院"UE4游戏开发"课程的学习,已经完成了UE4蓝图方面比较基础性的学习.通过UE4蓝图的开发,我实现了类似CS的单人版射击游戏,效 ...
- 体育游戏中的Player类
最近在做一个棒球的游戏,开始感觉还是挺酷炫的,但是其实做法挺朴实的,想象中的球员是多么智能,这样那样的,其实只是表象. 关于球员的类是游戏里非常重要的部分,这个玩意怎么写呢,可以这样写...... 棒 ...
- Aery的UE4 C++游戏开发之旅(3)蓝图
目录 蓝图 蓝图命名规范 蓝图优化 暴露C++至蓝图 暴露C++类 暴露C++属性 暴露C++函数 暴露C++结构体/枚举 暴露C++接口 蓝图和C++的结合方案 使用继承重写蓝图 使用组合重写蓝图 ...
- MMORPG大型游戏设计与开发(客户端架构 part16 of vegine)
由于近来比较忙碌和有些困倦的原因,所以关于这部分的文章没有及时更新,一句话:让朋友们久等了!今天所讲的是客户端vengine(微引擎)中最后一个部分,就像上节所说,这一部分的内容比较多.可能有些朋友看 ...
- MMORPG大型游戏设计与开发(客户端架构 part12 of vegine)
在游戏中的交互过程中输入是一个必不可少的过程,比如登陆的时候需要用户输入用户名与密码,就算是单机游戏很多时候也要求用户输入一个用户名作为存档的依据.网络游戏中没有了输入,只用鼠标来交互是不切实际的,因 ...
- MMORPG大型游戏设计与开发(客户端架构 part9 of vegine)
时间在人们的生活中是多么重要的东西,如果打乱了时间,不知道这个时间会成什么样子.在客户端中,自然也有时间模块,因为不同的时间可能会处理不同的事情,特别是在追求高度自由化的同时,时间也成为了一个很重要的 ...
- MMORPG大型游戏设计与开发(客户端架构 part3 of vegine)
无论在何处在什么地方,我们都或多或少的接触到数学知识.特别是在客户端中,从打开界面的那一刻起就有太多与数学扯上的关联,如打开窗口的大小,窗口的位置,窗口里面的元件对象,以及UI的坐标等等.而在进入游戏 ...
- AS3游戏中可视对象上限及位图相关的内存消耗实测
前些天连续做了一些测试,以加深对AS3的掌握和在项目中对 游戏 性能.效率优化方面的一些处理,有很多测试实际意义不大,都不过是证明一些猜想是正确的,除此没有什么. 但前天进行的一系列测试中,有一些对游 ...
随机推荐
- Git 创建分支并合并主分支
首先,我们创建dev分支,然后切换到dev分支: $ git checkout -b dev(等价于 $ git branch dev $ git checkout dev ) Switched to ...
- 配置DHCP中继
本实验模拟企业网络场景.某公司分部的网络由交换机S1和网关路由器R1组成,员工终端PC-1和PC-2都连接在S1上.公司要求分部内所有员工主机的IP地址都通过总部的DHCP服务器自动获取.分部网关路由 ...
- [19/06/07-星期五] CSS基础_布局&定位&背景样式
一.固定布局(不适应设备的浏览器的变化) <!DOCTYPE html> <html> <head> <meta charset="UTF-8&qu ...
- MySQL-快速入门(2)数据类型
1.MySQL支持多种数据类型: 1>数值类型:整数类型tinyint.smallint.mediumint.bigint.int:浮点小数类型float.double:定点小数类型decima ...
- HashMap中确定数组位置为什么要用hash进行扰动
HashMap数据存储的过程先根据key获得hash值,通过 (n - 1) & hash 判断当前元素存放的位置(这里的 n 指的是数组的长度),如果当前位置存在元素的话,就判断该元素与要存 ...
- media查询(来源于bootstrap)
/* 大屏幕 */@media (min-width: 1200px) { ... } /* 平板电脑和小屏电脑之间的分辨率 */@media (min-width: 768px) and (max- ...
- 100行代码撸完SpringIOC容器
用过Spring框架的人一定都知道Spring的依赖注入控制反转;通俗的讲就是负责实例化对象 和 管理对象间的依赖 实现解耦. 我们来对比两段代码: UserController{ UserServi ...
- 华为云搭建windows+wordpress+xampp
1.如何将本地文件上传至华为云ECS云服务器(Windows系统) 1.1 在本地电脑上,快捷键“WIN+R"打开“运行”中输入“mstsc”,点击确定 1.2 在“远程桌面连接”框点击“ ...
- 50. Pow(x, n) (JAVA)
Implement pow(x, n), which calculates x raised to the power n(xn). Example 1: Input: 2.00000, 10 Out ...
- nmblookup - 基于TCP/IP上的NetBIOS客户用于查询NetBIOS名字的程序
总览 SYNOPSIS nmblookup [-M] [-R] [-S] [-r] [-A] [-h] [-B <broadcast address>] [-U <unicast a ...