在FPS游戏中,保持游戏的同步性是非常重要的,因为游戏中的每个玩家都必须看到其他玩家的动作与行为,同时也需要感受到自己的行为在游戏中的影响。为了保持游戏的同步性,FPS游戏通常采用一种叫做“客户端-服务器”(client-server)的架构。在这种架构下,游戏会有一个服务器,用于管理游戏中的所有信息和事件,同时每个玩家的计算机也会作为客户端与服务器进行通信。当玩家在游戏中移动或进行其他动作时,这些动作会被发送到服务器进行处理。服务器将玩家的动作和行为记录下来,并将其广播给其他玩家的计算机。其他玩家的计算机会根据这些信息来更新他们的游戏画面和数据,以保持游戏的同步性。为了解决玩家延时不同的问题,FPS游戏中通常采用延迟补偿(lag compensation)技术。这种技术会根据不同玩家的延迟情况,对玩家的动作进行调整,以确保所有玩家在游戏中的动作都是准确的。除此之外,FPS游戏还会对网络传输进行优化,例如采用压缩技术来减少数据传输量,以降低游戏延迟。此外,游戏开发者也会对游戏进行测试和优化,以确保游戏在不同网络环境下都能保持良好的同步性。发布于 2023-02-24 10:15・IP 属地广东赞同添加评论分享收藏喜欢更多回答韦易笑游戏开发等 4 个话题下的优秀答主 关注网游技术发展真够慢的,怎么大家十年前在讨论同步问题,十年后还有人在讨论这个问题呢?何为延迟补偿?如何进行坐标差值?B客户端屏幕上A已经跑到东边了,但是收到服务器说“A正在西边往北跑”,B到底该何去何从?各位说了那么多DR,TimeWrap,LockStep,自己实现过没有?有产品上线没有?这些问题明明很简单,却总被形容得云山雾罩的,离落地编码还是那么的遥远。我若干年前的一个实现版本,将简明扼要的解决这个问题:--------------------------------------------------------------------------------------------------------------影子跟随算法由普通DR(dead reckoning)算法发展而来,我将其称为“影子跟随”意再表示算法同步策略的主要思想:屏幕上现实的实体(entity)只是不停的追逐它的“影子”(shadow)。服务器向各客户端发送各个影子的状态改变(坐标,方向,速度,时间)。各个客户端收到以后按照当前重新插值修正影子状态。影子状态是跳变的,但实体追赶影子是连续的,故整个过程是平滑的。前面的1号终端控制红色飞船P1向左飞,并把自己的状态时时告诉服务器
后面的2号终端上接收到飞船P1的影子S1的状态(向左移动),并让P1的实体追赶S1网络性能指标一:带宽,限制了实时游戏的人数容量网络性能指标二:延时,决定了实时游戏的最低反应时间使用该算法可以容易的开发出一款马里奥赛车,或者Counter Strike,详细说明见后:
算法比较:帧间同步:不同客户端每帧显示相同的内容,键盘/时钟数据传到服务器,服务器确认后所有终端做出响应,多用于局域网游戏,比如红警(需要等待客户端),街霸II的网络版(360),可参考 LockStep,TimeWrap算法,网速要求高,复杂度低,见我的旧文帧锁定算法。插值同步:不同客户端显示不同步,但是状态同步,常见的Dead Reckoning(或叫导航插值),效果好,但复杂度高。常见于竞速类游戏和 FPS游戏。算法定义:时间:单位为帧(FPS=10),开始由服务器告诉向所有客户端,每5分钟同步。玩家:每个玩家控制自己的实体,并在每贞将状态改变告知服务器。状态:状态数据 = 实体ID + 坐标 + 方向 + 速度 + 时间(贞)。插值:收到新状态包后将根据其运动方向与时间,根据现有时间计算新状态。跟随:实体不停的追踪自己的影子,追上后与影子保持状态同步。相位滞后:可选参数,实体与影子保持一定距离同步,相当于保持一定车距,这样在控制者突然停止的时候,不容易因为网络延迟跑过了又被拉回来。
惯性移动:可选参数,开始移动或者停止或者改变方向都有加速度,这样就不需相位滞后了。
每次服务器向各个客户端同步时间的时候,由于延迟,所有客户端的时间都是慢于服务器的,这没有关系,只要大家在一定误差范围内以相同的速度增加,就完全没有问题。
图2 IDC网络响应在公网平均130ms的Latency下,是不存在“完全的”的同步情况。如何通过消除/隐藏延时,将用户带入快速的交互式实时游戏中,体验完美的互动娱乐呢?
让所有的用户屏幕上面表现出完全不同的表象是完全没有问题的;
把这些完全不同表象完全柔和在一个统一的逻辑中也是完全没有问题的。
需要根据具体情况,分清楚哪些我们可以努力,哪些我们不值得努力,弄明白实时游戏中同步问题关键之所在,巧妙的化解与规避游戏,最终在适合普遍用户网络环境中(200ms),实现实时快速互动游戏。
案例解析:Counter Strike实现CS的话,首先我们需要给人物移动加上惯性,比如静止状态突然开始移动,那么需要0.5-1秒的加速过程,而移动中突然停止也需要0.5-1秒的减速过程,这样就实现了无差别同步,不需要相位滞后来避免拉扯影响用户感。
同时开枪射击采用客户端判断,也就是说如果我看见你在墙前面,开枪射中,那么我向服务器发送“我击中你了”,这时有可能真实的你在墙后,那么表现出来的就是我看见我打中你了(减不减血由服务段判断),而你没有看见我,觉得我穿墙打中你了。
图3 CS的同步逻辑关键状态进行缓存,不然如果别人向前连续跳五次,每次取得状态都取到最高点的话,别人客户端上的影子和跟随的实体会奇怪的持续的飞在天上,所以需要将起跳和落地这两个关键状态缓存,实体追赶时只有追上的第一个状态(一号影子)才能追逐第二个状态(二号影子)。
由此可以在完全时间同步的情况下平滑的跑动、跳跃,开枪射击采用客户端判断后手感得到提高,唯一需要担心的就是外挂,外挂多是实时游戏的代价,只能通过Cheating
Death等工具防止了。
案例解析:马里奥赛车用该算法实现马里奥赛车是很简单的,影子和实体都使用惯性,由于赛车惯性很大,不容易有突变的状态更新,所以效果会比FPS游戏更好。
玩家碰到道具后,马上在屏幕上隐藏该道具的显示并通知服务器,由服务器决断道具属谁,由于刚碰到道具就隐藏所以不会有碰到道具却在一段时间内无法取得延迟现象。
游戏道具系统实现也很容易,比如那个将当前第一名炸毁的道具,它的描述是:原角色+对象角色+约定发生时间。既然知道对象是谁,什么时间发生,那就更本不需要怎么同步了,所有客户端和服务器在该时间让炸弹爆炸就得了,这种手法类似即时战略游戏。
游戏还有一类道具是可以发射的乌龟壳,这个东西属于有弹道的发射物,类似Quake里面的某些武器,需要作一些同步处理,基本特性是服务器判断起决定作用,客户端同步判断,如果客户端与服务器都判断集中,那就集中;如果客户端判断集中而服务器判断没有集中,那会看到该角色似乎被打了一下,但很快又恢复了速度向前冲。
由于赛车本身就具备惯性比较大的特点,因此同步效果是比较好的,可以在更大的延迟情况下表现得和FPS差不多(比如300ms效果相当于FPS的200ms)。
非可靠包:该“影子跟随算法”支持非可靠传输协议,如果使用非可靠传输,那么我们按照特定频率(如每秒10次)定时发送状态更新,因为协议中每个更新包出了位置外还有速度、方向和时间,甚至还能加速加速度,因此我们丢一个包没有关系,可以根据后来的包重新计算插值。只有关键状态更新时才需要可靠传输,这就避免了TCP中丢包时RTO指数增长造成的延迟了。
负面情况:该算法缺点就是无法向“帧间同步”算法那样,每次发送按键给服务器,服务器处理后再反馈结果,在局域网中(平均延迟<5ms),这样的效果相当于单机游戏一样即时,游戏性也能很复杂。然而在Internet中(平均延迟130ms,设计基准200ms,每秒最多发送10个数据包)该算法却不能像单机游戏那样有复杂的场景互动,有类似格斗游戏的即时的动作判定。
许多策划在设计实时动作游戏时很多设计我们都难以实现,这样因为策划不容易明白哪些我们能做,哪些我们不能做。即便程序员精通同步理论,策划也经常碰壁。
当多数设计被程序员回复“无法实现”后,策划只有采取一种消极设计(砍掉很多有意思的互动元素),于是网络游戏的表现力到今天还是差单机游戏一大截。
这些问题也并不能因为“影子跟随算法”的提出而得到改进,大于100ms的判定时间,都很难做到即时。
最后,该算法编码复杂度比其他同步策略高,因为服务器需要计算一份影子数据,各个客户端需要计算一份影子数据,还需要计算实体追赶,而这三种计算都需要在同样的时钟下保持一致,这就增加了编码与调试的复杂度。
总结话题:Internet特点是“高带宽,高延迟”,可以说从本质上Internet就不是为了游戏而设计的。故此Internet绝对意义的同步是不存在的。“影子跟随算法”的核心思想有几个:时钟同步,客户端先行,平滑追赶。通过这三个特性,我们能够在近似时间同步的情况下,模拟各种物体的移动过程,而使用该算法的前提是设计者需要根据各个游戏的特性研究不同的优化技巧,策略因游戏而变。
比如发送状态更新包时,不需要每次都发送,而可以只发送改变的状态。什么时候我们觉得改变了?就是当客户端实体与自己的影子之间的误差大于某特定数值时我们才发送更新包,这样虽然玩家在原地做左右摇摆的小幅度移动,只要没有超出范围,都不需要发送新的状态更新,其他玩家机器上看起来,它是站着不动的。
比如当发现某客户端5秒钟没有相应了,那么就将该人物的影子冻结住,永远不要为了等待某个数据而不让游戏进行下去。
本算法需要客户端与服务器维护相同的时钟,当每5分钟同步的时候,直接根据服务器的时钟替换当前时钟就行了,不需要重新计算所有影子的位置,因为后续的状态数据将会马上刷新这些状态。更不需要将测量到的PING值考虑进去,该算法与PING具体值无关。
当发现策划案子不可行时,寻找近似替代方案,比如减少“一次性的”“决定性的”事件发生,比如延长导弹在空中飞行的时间,比如将敌人加入HP分多次打死,而不是以及毙命,等等,都是大家可以发挥想象的地方。
相关例子:文章相关DEMO如果有需要的话,可向我索要。
参考阅读:韦易笑(2006),帧锁定算法韦易笑(2005),网游同步法则展开阅读全文赞同 237461 条评论分享收藏喜欢腾讯天美工作室群游戏等 2 个话题下的优秀答主已关注840 人赞同了该回答同一局游戏内,每个客户端和服务器都各自是一个独立的世界。理想情况下,所有世界都应该是同步的、一致的,在任一给定时刻,玩家在每个世界中的位置和姿态都应该是相同的。然而由于延迟的存在,这种理想情况却不可能实现!尤其在射击游戏中,因为射击游戏给玩家的反应时间极短,1秒就能决定生死,职业玩家甚至对10毫秒的延迟都会非常敏感,所以在游戏中如果人物的移动忽快忽慢,甚至还有位置瞬移,玩家的射击体验就会大打折扣。今天我们邀请到了来自天美的游戏后台开发Walter,他将从技术角度分享自己解决延迟问题的经验:在射击游戏中,我们要如何与延迟抗争,做到更好的体验?1. 认识游戏中的延迟网游都是采用典型的C-S通信模式,客户端必须持续地与服务器通信才能正常运作,网络通信的round-trip time明显就是我们不想要的延迟。网络传输的丢包、重传等,都可以纳入网络延迟的范畴。但游戏中的延迟远不止网络延迟这么简单,计算机系统(包括手机)的运作、游戏的实现机制都会引入延迟。点击鼠标产生开火指令,这个电信号的传输、被系统捕获都需要时间,客户端逻辑线程通常要到下一帧才会开始处理这次输入,逻辑处理中抛事件到渲染线程,渲染线程最快也要下一帧才能去渲染结果,渲染结果要显示到屏幕上又受限于屏幕的刷新率。可见,即使是没有网络的纯客户端游戏,延迟都不可避免地存在。很多游戏在实现中会用到Buffering技术,这还会进一步加剧延迟。衡量游戏品质的一项重要指标就是玩家常说的“手感”,而手感的关键组成部分就是反馈的及时性。对于射击游戏、竞速游戏,玩家对手感有着极高的期待,职业玩家甚至能感知到10ms级别的延迟差距,所以游戏开发者都会在这方面下很大的功夫。2. 降低网络延迟网络游戏中,网络延迟是延迟的主体部分,降低网络延迟是优化体验的关键,也是开发商都会重视的地方。例如,由Riot Games开发的第一人称射击游戏《Valorant》就做到了为70%玩家提供小于35ms的RTT。2.1 搭建专用网络网络包在传输中要经过复杂的路由,这导致了高延迟以及更多的不确定性。所以,有实力的厂商都会在网络基础设施上做投入,例如Riot有去搭建自己的ISP。这一点可以简单理解为:厂商会围绕玩家就近部署很多接入服务器,内部的服务器之间由专用的高速通道相连,就像修了很多专用的高速公路,所以整体的通信非常稳定、高效。2.2 使用UDP越来越多的游戏都从TCP转向了UDP,在剔除/简化了拥塞控制以后,包传输的及时性有了明显的提升,尤其是在弱网环境下。由于UDP不可靠的特性,一般都会实现Reliable-UDP。实际游戏中,一般会混用可靠与不可靠的UDP,纯表现的消息、状态同步等可能会用不可靠的UDP,用户输入、逻辑事件等重要的信息则会用可靠的UDP。3. 使Buffer尽可能小3.1 Buffering机制网络游戏中,客户端和服务器几乎每一帧都在通信,玩家的操作永远来自于客户端,权威的信息永远来自于服务器。但网络通信是不稳定的,服务器可能连续5帧都没有收到客户端A的数,但在下一帧又一口气收到5个。这种输入的缺失和爆发增长会导致卡顿,导致不平滑的游戏体验。无论是帧同步还是状态同步,都会面对这个问题。经典的解决办法是使用buffering机制:增加一个buffer,缓存几帧数据以后再以稳定的频率向业务系统提供输入。这个机制良好地解决了网络传输带来的输入抖动问题,网络视频播放器中一直在使用该技术。3.2 缩小buffer很多游戏中都有用buffering机制,它确实让游戏变得平滑了,但却以增加延迟为代价。追求极致体验的游戏都不得不在buffer的大小上做折中,有的只存1帧的数据,有的甚至完全消除buffer。缩小buffer的同时还要保证平滑的游戏体验,这意味着游戏要提供非常稳定的帧率、避免毛刺,还必须要在没有输入的情况下自己去预测玩家的行为。假设玩家在持续地移动,客户端每帧都在上报移动指令,后来由于网络故障,服务器在几帧内都未收到新的移动指令,这时服务器要尽最大的努力去预测,从而使网络恢复以后 玩家的移动仍然是平滑的。4. 更高的帧率帧率的影响到底大不大?帧率高到一定值以后 是否值得继续提高?这是一个有一定争议的话题。油管上有一个高帧率对玩家表现的对比测试,感兴趣的同学可以观看一下。视频地址::https://youtu.be/OX31kZbAXsA 不管是否值得,但毫无疑问,更高的帧率必然是有好处的。帧率越高,系统运作的延迟就越低,响应越及时,表现也会更平滑、更自然。客户端帧率超过100的游戏并不少见,但服务器帧率几乎都小于64。服务器的帧率有必要更高吗?Riot说有必要,《Valorant》的服务器是以128帧运行的,这让人印象深刻。客户端与服务器都在模拟角色的移动,它们的运算结果会无可避免地出现分歧。假设服务器是1秒1帧,两端位置的最大分歧是1米,然后被强制纠正到一致。若两端都是1秒128帧,两端位置的最大分歧可能就只有10厘米了,然后也被强制纠正到一致。帧率高了就像是小步快跑,每次产生很小的误差,然后不断地纠正,所以整体的表现会很平滑。另一方面,即使某个客户端只以60帧的频率上报移动指令,其他9个客户端仍然可以以128帧的频率收到服务器下发的位置更新,所以 其他9个人看到的人物移动仍然是非常平滑的。在《Valorant》中,客户端与服务器的移动和物理都是保持128帧运行的,这使得两端的每一帧都可以 一一 对应上,这极大地方便了服务端的命中判定,服务器只需把其他人的位置回退固定的帧数,就可以与客户端上的位置匹配。《Valorant》高帧率缩短了系统运作的延迟,提供了更加平滑、细腻且一致的角色移动,这对FPS游戏是很有意义的。射击游戏给玩家的反应时间是极短的,1秒就决定了生死,职业玩家对10毫秒的延迟都很敏感。每个人都是非常平滑地移动,这是很利于玩家追踪和预测对方移动轨迹的,所以玩家的射击体验会好很多。要是人物的移动忽快忽慢,甚至还有位置瞬移,玩家的射击体验就会大打折扣。但是,一切皆有代价。要做到这样的高帧率本身就很困难,需要花很大的力气去优化性能。就算技术上做到了,服务器运行的成本也是非常高的。5. 状态同步+客户端预测+纠正帧同步要求每个客户端的运算输入严格一致,这样才能保证每个世界始终处于一样的状态。这意味着它们的输入一定来自于同一个服务器,意味着玩家输入到逻辑执行之间要包含网络延迟,这个延迟通常是几十毫秒,这会立即让玩家觉得“游戏的手感不好”。FPS游戏大多都是使用状态同步,因为这样很容易做客户端预测,从而实现更好的及时性。玩家按键操作之后,客户端把该输入发给服务器,同时客户端本地直接处理该输入、产生部分结果/表现,而不用等待服务器的返回,这就是客户端预测。服务器处理完输入以后会通知客户端结果,如果本地运算结果与服务器下发的不一致,肯定要以服务器的为准,故客户端要做纠错处理。由于是状态同步,即使客户端本地计算出错了,也总是能恢复到一致的、正确的状态,UE引擎的移动同步就是这样做的。6. 隐藏lag对于网络游戏,重要的信息必须以服务器为准、不能客户端自主决定,否则会外挂泛滥,例如子弹命中的判定、造成的伤害量、技能的生效等等。但如果玩家的输入总是要等服务器返回后才给玩家反馈的话,玩家会觉得游戏的体验很差。公平性与及时性都很重要,但却存在冲突,需要我们做一些细致的特殊处理。6.1 扔手雷接下来,我们来看一个实际的例子(此处的例子节选自由微软发行的第一人称射击游戏系列《光环(Halo)》的分享)。假设玩家扔手雷的粗略过程如下:按键是客户端行为,前摇动画可以看作是一段固定的延迟,飞出手雷代表施法的正确结束、产生了效果。我们是网游,生成手雷这么重要的事情当然得服务器说了算,所以我们会很自然地这样做:图中黄色的文字标记出了延迟的位置,玩家按键以后 隔一段时间才出现前摇动画,这体验当然是不可接受的,我们要更早地给玩家反馈。想要及时反馈,可以在按键后直接播前摇动画,并且在时间结束后直接飞出手雷:这违背了服务器做决策的原则,明显更无法接受。服务器上前摇过程还未走完,但客户端已经飞出手雷了,若服务器上前摇过程被他人打断,客户端必须要把已经飞出的手雷给删掉,这种诡异的表现会让玩家非常困惑和不满。《Halo》最终的实现方案是:按键后,客户端立即播放前摇动画,但等服务器流程结束后再通知客户端飞出手雷。相当于在客户端把前摇给拉长了。这样做解决了上述问题,提供了良好的用户体验:● 玩家按键后立即获得了反馈,所以操作体验很好。● 手雷的生成遵从了服务器的权威,杜绝了被打断导致的奇怪表现。● 虽然手雷的飞出有延迟,但动画末尾的手臂占据了大部分画面,玩家几乎感知不到这个延迟。即使玩家发现了这个延迟,影响也不大。6.2 开启无敌另外一个例子,是《Halo》中释放无敌技能的过程。我们仍然期望按键后就立即播放前摇动画,但却不能再像上面一样延迟出现无敌效果。因为无敌状态对施法者太重要了,晚0.1秒就可能是生与死的差别,玩家对这段延迟非常在意、几乎无法容忍。这时还可以把延迟藏在哪呢?为了给施法者提供极致的体验,这里修改了游戏机制 —— 把服务器端的施法延迟给缩短了,即把网络延迟藏在了这里。这种修改游戏机制的做法是特例,应该谨慎使用。虽然施法方的体验变好了,但对他的对手是不公平的。这里之所以可以这么做,是因为玩家普遍不能接受自己的无敌晚一点出现,但却相对能容忍对方的无敌早一点出现。7. 在割裂的世界中愉快地玩耍7.1 屏幕上看到的都是假象我们再来看看射击游戏中的经典问题 —— 命中判定。延迟的存在,使这个问题变得更加棘手。子弹是否打中敌人、打中的部位是头还是脚,这些都是非常关键的信息。命中的判定可以在客户端做,然后服务器做校验,也可以只在服务器做。无论用哪种方式,我们都必须要面对一个事实:玩家屏幕上看到的敌人位置,很可能不是此刻敌人的真实位置,或者说不是其他世界中对方的位置。同一局游戏内,每个客户端和服务器都各自是一个独立的世界。理想情况下 所有世界都应该是同步的、一致的,在任一给定时刻,玩家A在每个世界中的位置和姿态(Pose)都应该是相同的。但由于延迟的存在,这种理想情况永远都不可能实现!一个简化后的、现实中的模拟如下图:一个框代表游戏运行的一帧,图中只画出了部分帧,并假设客户端与服务器的逻辑帧能稳定地一一对应。红线代表客户端上报的自己当前帧的位置/移动,蓝线代表服务器每帧下发的其他角色的位置/移动。假设A和B都在持续地移动:l t1时刻,客户端A把自己本帧移动的结果(PAe)上报给服务器,客户端B同样把PBe上报给服务器。l t2时刻,服务器上A和B的位置分别为PAe和PBe。服务器把当前帧的位置信息同步给所有客户端。l t3时刻,客户端A收到玩家B的位置为PBe,但自己已经移动到了PAi。 这是很多游戏的角色位置模拟过程,在任一时刻,客户端看到的自己的位置始终是领先于服务器的,但看到的其他人的位置又都是落后于服务器的。这导致——战斗中,我们始终是在用将来位置的自己去打过去位置的敌人。例如客户端A在t3时刻瞄准了B,并开枪射击,服务器收到射击事件时已经是第i帧了,但t3时刻和t4时刻 玩家B在服务器上的位置都不是PBe。假设命中判定在客户端做,t3时刻客户端A认为自己打中了B,但服务器需要校验这一次射击的合法性,服务器收到射击事件时 玩家B的位置已经变成PBi了,这时服务器很可能认为A打不中B。把命中判定交给服务器来做,仍然有同样的问题。不同世界间的不同步是由延迟导致的,延迟导致的结果是:只要敌人持续在移动,并且移动速度比较快,你瞄准后的射击永远也打不中敌人。7.2 服务器回退玩家的位置解决命中问题的经典方案是 —— 服务器做玩家位置回退。服务器记录一段时间内每个人的历史位置和Pose,收到A的射击事件时,服务器把A之外的所有玩家都挪回到一个恰当的历史位置去,使服务器上其他人的位置与A看到的其他人位置基本一致。这样,客户端上能打中的 服务器上就也能打中了。如图所示:但这种方案也有问题:它对被打的那一方不公平。例如t3时刻,玩家B明明已经躲起来了,在客户端B的世界里 玩家A是打不到自己的(位置PAe与位置PBi),结果你t3时刻的开火还是打死了自己。要解决这个问题,一般是给回退的时长加一个约束,例如最多只允许回退一个RTT。8. 让网络卡的人自己拉扯UE4默认的移动同步过程是:客户端和服务器都模拟角色的移动,客户端每帧上报自己移动后的结果和当前的移动输入,服务器收到移动输入后也去模拟移动。若服务器的移动结果与客户端上报的结果误差在一定容忍范围内,则直接采纳客户端的结果,使服务器上玩家的位置等于客户的上报的位置。若误差超出了容忍范围,则以服务器为准,强制重置客户端的位置。当容忍范围设置得较大的时候,服务器上会大幅度重置角色的位置,当同步到其他客户端以后,这个角色的移动必然会出现波动,要么是直接位置重置,要么是插值过去(加速移动过去)。无论用哪种处理方式,最终的结果都是:一个玩家B卡了,然后其他9个人看到的B的移动都是不平滑的。Valorant认为:一个人卡了,却导致其他9个人的瞄准体验变差了,这是不可接受的。对应方案是:若服务器未收到移动输入,仍然根据当前的状态每帧做预测,当出现分歧时始终以服务器的模拟为准,并且每帧都下发角色的位置。这样做,无论个人怎么卡,其他9个人看到的角色移动始终是平滑的,但卡的那个人自己会被频繁地拉扯。9. 解决Peeker’s advantage9.1 什么是Peeker’s advantage?这是战术射击游戏的经典话题,玩家在转角处来回晃悠可以获得先手优势(会先看到对方),是一种网络延迟导致的不公平现象。如图所示,冲出角落的人会先看到对方,而蹲角落的人明显要晚一些看到对方。说明:这里是用了比较大的ping值来演示该效果,实际中不一定有这么明显。产生这个问题的本质原因就是网络延迟,正如上面已经讲过的:若是自己在移动,自己客户端上自己的位置是领先于其他所有世界的,而静止不动的人的位置在所有世界中都是一致的。移动的人的位置要同步到其他客户端,这是需要经过网络传输的:9.2 它到底有多大的影响?这个优势其实是可以用数学公式来描述的:如图,holder要想打赢对方,就必须要在server applies kill shot之前把开火指令发送到服务器,这样才能实现比peeker先开火。进而可以得到这样的关系:holder reaction time < peeker reaction time - (holder RTT + holder buffering latency + server buffering latency)所以peeker方领先的时间主要取决于holder方的网络延迟,以及buffering延迟。更进一步,我们可以带入具体的数字来量化这个优势。按帧率60、单边通信30ms延迟来算,peeker方大约可以获得140ms的先手优势。9.3 技术层面能做什么?① 降低网络延迟。无非就是就近部署、搭建高速专用网络、使用UDP,上面已经讲过了。② 降低buffering延迟。最小化buffer大小,上面也已经讲过了。③ 高帧率也是有帮助的。在Valorant中,如果客户端能跑140帧的话,这个差值可以降低到71ms。9.4 从设计的角度弱化Peeker’s advantage技术上没办法彻底杜绝这个问题,而且也很难取得明显的改善,至少代价是很大的。但从设计上 我们有很多方法来优化它:● 地图设计:例如一个区域只留一两个进入的入口(供他人peek around),但场地内有多个地方可以设伏、蹲守入口。这样的设计使得holder的位置高度不确定,但peeker的位置高度确定,变相地增加了peeker方的难度。● 降低移动中射击的准确率。在这一点上每个游戏的取舍都不同,Valorant在这方面做得比较狠,移动中几乎很难打中。● 让peeker的身体先暴露,但相机晚一点暴露。这使得peeker先暴露出自己的部分身体,然后相机才能看到holder。人的移速慢一点、系统运行帧率高一点、身体做大一点,都会有利于peeker先暴露出部分身体。出处和参考: