MMO技能系统的同步机制分析

转自:http://www.gameres.com/729629.html

此篇文章基于之前文章介绍的技能系统,主要介绍了如何实现MMO中的技能系统的同步。阅读此文章之前,推荐首先阅读前一篇文章:一个MMORPG的常规技能系统

  这里所说的技能系统包括:技能流程和技能创生体(法术场、弹道和buff)。

  首先介绍authority和proxy的概念,这两个概念是基于单位unit的基础上进行的区分。

  authority表示单位的主控端,即此单位是由客户端和还是服务端控制。对于玩家avatar,玩家本地的客户端就是主控端。而对于怪物,他们的行为由服务端控制,主控端就是服务端。

  proxy表示代理端,表示被主控端控制。如对于怪物来说,所有的客户端都是proxy;对于玩家A来说,服务端和其他玩家的客户端都是proxy。

  0 技能同步的原则

  1.客户端先行

  对于玩家控制的单位来说,玩家点击按钮释放一个技能,客户端首先响应,单位播放动作以及相应的技能特效。

  据我了解,有的已上线游戏并没有做客户端先行,而是所有的技能执行请求都发给服务端,然后由服务端发起。

  这种模式技能流程控制会比较简单,但是在网络环境差的情况下,体验可能差一些。但是,目测也是可以接受的。

  2.技能流程以authority为发起端

  玩家单位技能发起是由她的客户端,怪物的技能发起是由AI也就是服务端。

  3.技能结算在服务端发起。

  技能真正的结算,比如法术场检测、buff结算、伤害计算以及扣血等,统一在服务端处理。

  1 技能执行流程的同步

  这里所说的技能执行流程指的是技能树的一个执行节点的流程。

  技能流程负责动作、特效以及技能结算,其中技能结算包括:释放法术场、弹道或buff。

  一个技能执行节点的执行流程中,需要同步的有两个时间点:

  • 技能开始:技能开始播放动作
  • 技能结算:前摇结束,即能进入结算逻辑。这类同步消息往往并不是由技能本身去同步,而是技能生成了法术场、弹道等,他们去做相应的同步。

  以玩家点击技能按钮开始释放技能为例介绍技能同步流程,如图所示:

<ignore_js_op>

  1.主控端点击技能按钮,技能开始播放动作,主控端告诉服务端技能开始。

  2.服务端广播给所有的客户端(多玩家场景),告知其他所有的客户端此玩家开始执行技能。其他客户端收到指令后可是播放技能表现。

  3.服务端延迟一段时间后,服务端开始进行技能结算,并且将结算结果通知客户端。

  延迟时间=技能前摇时间-上行-下行,下行一半不能确定,所以默认为上行=下行

  另一种中庸的计算方式是:延迟时间=技能前摇时间-上行,防止要求技能前摇时间过长

  使用此同步流程的表现为:

  1.要求技能前摇时间>2*网络延迟,若前摇时间短,则延迟时间=0,效果可能差一些

  2.authority客户端表现完美。

  3.proxy client表现一般,即玩家A看玩家B的效果为:玩家B刚开始执行技能动作,没到前摇时间就进行了技能结算。但是因为玩家一般也不会过分关注其他玩家的动作,所以是可以接受的。

  2 技能树的同步

  我上篇文章 一个MMORPG的常规技能系统 已经介绍,我们游戏使用的是技能树来管理技能流程。那么就面临一个问题,技能树如何同步。

  最简单最暴力的方式,是客户端和服务端同时管理技能树,并且将其状态同步。这样,客户端和服务端的技能树状态统一、完备。

  后来发现,对于proxy端,并不需要完备的技能树信息,最节省的方式是proxy根本不接受技能树同步信息,只是接受播放动作、技能结算等信息。但这样需要告诉其他proxy播放什么动作、特效等。

  在我们系统中,技能同步包括三类同步消息:

  • 技能根节点enter (root_enter): 表示一个大技能的进入。
  • 技能叶子节点enter(action_enter): 表示一个技能树的执行节点的进入。
  • 根节点exit(root_exit) :表示大技能结束。

  根节点保存一个完整技能的信息,需要和技能模块外部交互,因此需要知道技能的开始和结束。

  叶子节点的执行代表着技能真正的执行逻辑,也需要同步。

  而对于其他节点,作为流程控制节点,只需要在主端确保技能流程无误即可

  后来这里进行了进一步的优化,对于纯根节点,主控端(玩家控制的客户端)将信息同步给服务端,服务端不再同步给其他客户端。有的技能树只有一个节点,那么按照叶子节点的策略,主控端同步给服务端,服务端广播给所有的其他客户端。

  3 技能结算的同步

  技能结算包括创建法术场、buff、弹道、技能直接伤害等。

  法术场、弹道的同步

  法术场、弹道的同步比较类似,他们都作为一个entity(网络同步单元)在服务端创建,创建以后使用entity管理机制服务端通知客户端他们的创建和销毁。

  以法术场为例,法术场的执行和同步流程:

  • 服务端发起创建一个法术场,并且通知客户端。
  • 法术场每隔一段时间结算一次,注意,法术场结算并不需要同步,每隔一段时间服务端执行检测逻辑,客户端播放结算特效等。两个逻辑互不依赖,也不要求时间一致。
  • 当法术场结算时检测到攻击目标时,服务端计算攻击伤害等信息,并将攻击信息发给客户端。
  • 客户端收到伤害信息,客户端播放相应的表现,如法术场受击特效等。此处还包括属于通用模块的跳字等。
  • 当服务端的法术场时间到了进行destroy时,使用entity的管理机制通知所有客户端destroy法术场。

  弹道的同步类似,唯一的区别就是法术场在某一位置使用攻击盒检测目标,而弹道是一个移动的子弹,客户端表现是一个特效在飞,而服务端每隔一段时间根据飞行速度等使用胶囊攻击盒去检测目标碰撞。

  由以上可以发现,法术场作为一个entity他的管理成本是比较高的,所以若策划想出一些需求需要使用多个法术场实现,一般通过拓展法术场功能使用一个法术场来实现。

  比如,策划要做一个冰火两重天法术场,即法术场在每次结算时使用不同的参数,第一次结算使用火焰,第二次结算是冰霜。若这种需求较少,可以使用两个法术场,但是如果要冰火雷毒水电风魔奥术神圣*N重天,则代价太大。一般可以让法术场支持每次使用不同的结算参数来结算即可。

  buff同步

  buff是附加在unit身上的东西(没有unit就没有buff,但是没有unit可能有法术场),所以不需要使用entity来同步。

  • 服务端确定buff是否可以挂在unit上面。
  • 客户端和服务端都维护一个buff管理器,挂buff的消息通知所有客户端,客户端负责表现,服务端负责结算即可。

  4 伤害、属性的同步

  主要介绍下伤害的同步,顺便附带介绍下属性同步。

  对于伤害结算来说,技能、buff、法术场和弹道都可能造成伤害,当服务端发现造成伤害时,服务端首先根据技能信息计算伤害值,计算以后将伤害信息发送给所有客户端,所有客户端接到信息后首先播放技能伤害相关的表现,如受击特效等,然后播放跳字等通用伤害客户端表现。

  也就是说,伤害值的同步其实就是简单的rpc消息。

  buff可能修改单位属性,如攻击力、攻击速度等。这些属性值的同步一般使用属性同步,属性同步的意思是当一个值改变了,底层自动把这个值同步给客户端。

  血量等信息值都是用属性同步方式。

  伤害计算和属性模块是一个重要且比较复杂的模块(当然,若游戏数值非常简单也可以很简单),后面我会写文章详细介绍这个模块。

时间: 2024-10-11 13:35:23

MMO技能系统的同步机制分析的相关文章

ffmpeg转码MPEG2-TS的音视频同步机制分析

http://blog.chinaunix.net/uid-26000296-id-3483782.html 一.FFmpeg忽略了adaptation_field()数据FFmpeg忽略了包含PCR值的adaptation_filed数据; 代码(libavformat/mpegts.c)分析如下: /* 解析TS包 */int handle_packet(MpegTSContext *ts, const uint8_t *packet){  ...   pid = AV_RB16(packe

.NET Core采用的全新配置系统[10]: 配置的同步机制是如何实现的?

配置的同步涉及到两个方面:第一,对原始的配置文件实施监控并在其发生变化之后从新加载配置:第二,配置重新加载之后及时通知应用程序进而使后者能够使用最新的配置.要了解配置同步机制的实现原理,先得从认识一个名为ConfigurationReloadToken的类型开始. [ 本文已经同步到<ASP.NET Core框架揭秘>之中] 目录一.从ConfigurationReloadToken说起二.Configuration对象与配置文件的同步三.应用重新加载的配置四.同步流程总结 一.从Config

从Handler+Message+Looper源代码带你分析Android系统的消息处理机制

PS一句:不得不说CSDN同步做的非常烂.还得我花了近1个小时恢复这篇博客. 引言 [转载请注明出处:http://blog.csdn.net/feiduclear_up CSDN 废墟的树] 作为Android开发人员,相信非常多人都使用过Android的Handler类来处理异步任务. 那么Handler类是怎么构成一个异步任务处理机制的呢?这篇 博客带你从源代码分析Android的消息循环处理机制.便于深入的理解. 这里不得不从"一个Bug引发的思考"開始研究Android的消息

MySQL系列:innodb引擎分析之线程并发同步机制

innodb是一个多线程并发的存储引擎,内部的读写都是用多线程来实现的,所以innodb内部实现了一个比较高效的并发同步机制.innodb并没有直接使用系统提供的锁(latch)同步结构,而是对其进行自己的封装和实现优化,但是也兼容系统的锁.我们先看一段innodb内部的注释(MySQL-3.23): Semaphore operations in operating systems are slow: Solaris on a 1993 Sparc takes 3 microseconds (

从Handler+Message+Looper源码带你分析Android系统的消息处理机制

引言 [转载请注明出处:从Handler+Message+Looper源码带你分析Android系统的消息处理机制 CSDN 废墟的树] 作为Android开发者,相信很多人都使用过Android的Handler类来处理异步任务.那么Handler类是怎么构成一个异步任务处理机制的呢?这篇 博客带你从源码分析Android的消息循环处理机制,便于深入的理解. 这里不得不从"一个Bug引发的思考"开始研究Android的消息循环处理机制.说来话长,在某一次的项目中,原本打算开启一个工作线

Android系统进程间通信Binder机制在应用程序框架层的Java接口源代码分析

文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6642463 在前面几篇文章中,我们详细介绍了Android系统进程间通信机制Binder的原理,并且深入分析了系统提供的Binder运行库和驱动程序的 源代码.细心的读者会发现,这几篇文章分析的Binder接口都是基于C/C++语言来实现的,但是我们在编写应用程序都是基于Java语言的,那么,我 们如何使用Java语言来使用系统的Binder机

分析.Net里线程同步机制

我们知道并行编程模型两种:一种是基于消息式的,第二种是基于共享内存式的. 前段时间项目中遇到了第二种 使用多线程开发并行程序共享资源的问题 ,今天以实际案例出发对.net里的共享内存式的线程同步机制做个总结,由于某些类库的应用属于基础,所以本次不对基本使用做出讲解,基本使用 MSDN是最好的教程. 一.volatile关键字      基本介绍: 封装了 Thread.VolatileWrite() 和  Thread.VolatileRead()的实现 ,主要作用是强制刷新高速缓存.     

深入理解Windows系统——3.Windows同步机制

windows系统在内核态,用户态,提供了很多种同步机制,用于控制都处理器.多线程并发. 高IRQL的同步 互锁操作 自旋锁 排队的自旋锁 栈内排队自旋锁 执行体的互锁操作 低IRQL的同步 互斥体(仅用于内核模式) 互斥体(被导出至用户模式) 信号量 事件 事件对 定时器 这些同步方式,有些适用于内核模式,如:写驱动的时候,有些适用于用户模式,如:写应用程序的时候,这里不再详细赘述. 原文地址:https://blog.51cto.com/14207158/2369084

Chromium Graphics: GPU客户端之间同步机制的原理和实现分析-Part I

摘要:Chromium中GPU进程架构导致多个GPU客户端会同时访问GPU服务,而多个GPU客户端可能存在数据依赖关系,例如渲染WebGL页面时,因此需要提供一种同步机制保证GPU操作的先后次序.本文讨论的就是多进程架构下GPU客户端之间的同步问题,以及同步点(SyncPoint)机制的基本原理. GPU进程架构等基本概念 我们知道,Chromium是一个多进程架构的软件系统.出于安全和稳定性方面的考虑,Chromium有个专门的进程(或者线程)和GPU设备进行交互,执行GL操作,也就是说,任何