大型多人在线游戏服务器架构设计 - RPC封装原理

RPC框架分为客户端部分与服务端部分:


RPC-client的部分又分为:

(1)序列化反序列化的部分(上图中的1、4)

(2)发送字节流与接收字节流的部分(上图中的2、3)

前一篇文章讨论了序列化与范序列化的细节,这一篇文章将讨论发送字节流与接收字节流的部分。

客户端调用又分为同步调用与异步调用

同步调用的代码片段为:

Result = Add(Obj1, Obj2);// 得到Result之前处于阻塞状态

异步调用的代码片段为:

Add(Obj1, Obj2, callback);// 调用后直接返回,不等结果

处理结果通过回调得到:

callback(Result){// 得到处理结果后会调用这个回调函数

        …

}

这两个调用方式,RPC-client里,处理方式也不一样,下文逐一叙述。

RPC-client同步调用

所谓同步调用,在得到结果之前,一直处于阻塞状态,会一直占用一个工作线程,上图简单的说明了一下组件、交互、流程步骤。

上图中的左边大框,就代表了调用方的一个工作线程。

左边粉色中框,代表了RPC-client组件。

右边橙色框,代表了RPC-server。

蓝色两个小框,代表了同步RPC-client两个核心组件,序列化组件与连接池组件。

白色的流程小框,以及箭头序号1-10,代表整个工作线程的串行执行步骤:

1)业务代码发起RPC调用,Result=Add(Obj1,Obj2)

2)序列化组件,将对象调用序列化成二进制字节流,可理解为一个待发送的包packet1

3)通过连接池组件拿到一个可用的连接connection

4)通过连接connection将包packet1发送给RPC-server

5)发送包在网络传输,发给RPC-server

6)响应包在网络传输,发回给RPC-client

7)通过连接connection从RPC-server收取响应包packet2

8)通过连接池组件,将conneciont放回连接池

9)序列化组件,将packet2范序列化为Result对象返回给调用方

10)业务代码获取Result结果,工作线程继续往下走

RPC框架需要支持负载均衡、故障转移、发送超时,这些特性都是通过连接池组件去实现的。

连接池组件


典型连接池组件对外提供的接口为:

int ConnectionPool::init(…);

Connection ConnectionPool::getConnection();

intConnectionPool::putConnection(Connection t);

【INIT】

和下游RPC-server(一般是一个集群),建立N个tcp长连接,即所谓的连接“池”

【getConnection】

从连接“池”中拿一个连接,加锁(置一个标志位),返回给调用方

【putConnection】

将一个分配出去的连接放回连接“池”中,解锁(也是置一个标志位)

如何实现负载均衡?

回答:连接池中建立了与一个RPC-server集群的连接,连接池在返回连接的时候,需要具备随机性。

如何实现故障转移?

回答:连接池中建立了与一个RPC-server集群的连接,当连接池发现某一个机器的连接异常后,需要将这个机器的连接排除掉,返回正常的连接,在机器恢复后,再将连接加回来。

如何实现发送超时?

回答:因为是同步阻塞调用,拿到一个连接后,使用带超时的send/recv即可实现带超时的发送和接收。

总的来说,同步的RPC-client的实现是相对比较容易的,序列化组件、连接池组件配合多工作线程数,就能够实现。还有一个问题,就是【“工作线程数设置多少最为合适?”】,这个问题在之前的文章中讨论过,此处不再深究。


RPC-client异步回调


所谓异步回调,在得到结果之前,不会处于阻塞状态,理论上任何时间都没有任何线程处于阻塞状态,因此异步回调的模型,理论上只需要很少的工作线程与服务连接就能够达到很高的吞吐量。

上图中左边的框框,是少量工作线程(少数几个就行了)进行调用与回调。

中间粉色的框框,代表了RPC-client组件。

右边橙色框,代表了RPC-server。

蓝色六个小框,代表了异步RPC-client六个核心组件:上下文管理器,超时管理器,序列化组件,下游收发队列,下游收发线程,连接池组件。

白色的流程小框,以及箭头序号1-17,代表整个工作线程的串行执行步骤:

1)业务代码发起异步RPC调用,Add(Obj1,Obj2, callback)

2)上下文管理器,将请求,回调,上下文存储起来

3)序列化组件,将对象调用序列化成二进制字节流,可理解为一个待发送的包packet1

4)下游收发队列,将报文放入“待发送队列”,此时调用返回,不会阻塞工作线程

5)下游收发线程,将报文从“待发送队列”中取出,通过连接池组件拿到一个可用的连接connection

6)通过连接connection将包packet1发送给RPC-server

7)发送包在网络传输,发给RPC-server

8)响应包在网络传输,发回给RPC-client

9)通过连接connection从RPC-server收取响应包packet2

10)下游收发线程,将报文放入“已接受队列”,通过连接池组件,将conneciont放回连接池

11)下游收发队列里,报文被取出,此时回调将要开始,不会阻塞工作线程

12)序列化组件,将packet2范序列化为Result对象

13)上下文管理器,将结果,回调,上下文取出

14)通过callback回调业务代码,返回Result结果,工作线程继续往下走

如果请求长时间不返回,处理流程是:

15)上下文管理器,请求长时间没有返回

16)超时管理器拿到超时的上下文

17)通过timeout_cb回调业务代码,工作线程继续往下走

上下文管理器

为什么需要上下文管理器?

回答:由于请求包的发送,响应包的回调都是异步的,甚至不在同一个工作线程中完成,需要一个组件来记录一个请求的上下文,把请求-响应-回调等一些信息匹配起来。

如何将请求-响应-回调这些信息匹配起来?

这是一个很有意思的问题,通过一条连接往下游服务发送了a,b,c三个请求包,异步的收到了x,y,z三个响应包:


(1)怎么知道哪个请求包与哪个响应包对应?

(2)怎么知道哪个响应包与哪个回调函数对应?

回答:这是通过【请求id】来实现请求-响应-回调的串联的。



整个处理流程如上,通过请求id,上下文管理器来对应请求-响应-callback之间的映射关系:

1)生成请求id

2)生成请求上下文context,上下文中包含发送时间time,回调函数callback等信息

3)上下文管理器记录req-id与上下文context的映射关系,

4)将req-id打在请求包里发给RPC-server

5)RPC-server将req-id打在响应包里返回

6)由响应包中的req-id,通过上下文管理器找到原来的上下文context

7)从上下文context中拿到回调函数callback

8)callback将Result带回,推动业务的进一步执行

如何实现负载均衡,故障转移?

回答:与同步的连接池思路相同。不同在于,同步连接池使用阻塞方式收发,需要与一个服务的一个ip建立多条连接,异步收发,一个服务的一个ip只需要建立少量的连接(例如,一条tcp连接)。

如何实现超时发送与接收?

回答:同步阻塞发送,可以直接使用带超时的send/recv来实现,异步非阻塞的nio的网络报文收发,如何实现超时接收呢?(由于连接不会一直等待回包,那如何知晓超时呢?)这时,超时管理器就上场啦。

超时管理器



超时管理器,用于实现请求回包超时回调处理。

每一个请求发送给下游RPC-server,会在上下文管理器中保存req-id与上下文的信息,上下文中保存了请求很多相关信息,例如req-id,回包回调,超时回调,发送时间等。

超时管理器启动timer对上下文管理器中的context进行扫描,看上下文中请求发送时间是否过长,如果过长,就不再等待回包,直接超时回调,推动业务流程继续往下走,并将上下文删除掉。

如果超时回调执行后,正常的回包又到达,通过req-id在上下文管理器里找不到上下文,就直接将请求丢弃(因为已经超时处理过了)。

however,异步回调和同步回调相比,除了序列化组件和连接池组件,会多出上下文管理器,超时管理器,下游收发队列,下游收发线程等组件,并且对调用方的调用习惯有影响(同步->回调)。异步回调能提高系统整体的吞吐量,具体使用哪种方式实现RPC-client,可以结合业务场景来选取(对时延敏感的可以选用同步,对吞吐量敏感的可以选用异步)。


本文转自架构师之路,作者:沈剑,转自游戏技术网:http://www.youxijishu.com/nd.jsp?id=127&_np=2_323

游戏技术网公众号,扫描加入讨论游戏技术


### 回答1: 网络多人游戏架构与编程是一本关于网络游戏开发的重要参考书籍,涵盖了从游戏架构设计到编程实现的全面内容,可帮助读者掌握网络游戏的开发技术和实践经验。 本书主要分为三部分,第一部分介绍了游戏架构的设计和实现,主要包括游戏服务器、客户端和网络通信的相关概念、原理和实现方法,以及游戏中的各种角色、场景、物品等的设计与实现。 第二部分则重点讲解游戏服务器的编程实现,从数据库设计、网络通信、多线程编程、内存管理、性能优化等方面详细介绍了服务器端的开发技术和注意事项,帮助读者掌握网络游戏服务器端的实现方法和技术。 第三部分则涵盖了游戏客户端的开发技术与实践,包括游戏客户端的编程架构设计游戏逻辑实现、动画效果、界面设计、音效处理等方面的技术,帮助读者掌握网络游戏客户端端的实现技术和设计思路。 总之,网络多人游戏架构与编程对于网络游戏爱好者、开发者和研究者来说是一本非常重要的参考书籍,其详实的内容与严谨的理论分析能够帮助读者深入了解和掌握网络游戏的开发与实现技术。读者可以通过下载PDF格式的书籍进行学习和研究。 ### 回答2: 网络多人游戏架构与编程 pdf下载是一本介绍网络游戏开发的电子书籍,涵盖了多方面的内容,如服务器架构、多人游戏的通信协议、客户端和服务器端的开发等。 在这本书中,作者详细介绍了传统的网络游戏开发方式,并提出了一种更为高效的架构设计思路,让开发者们能够在开发多人游戏时更加快速和高效。 网络游戏开发需要有一定的编程基础,因为需要通过编写代码来实现游戏的各种功能。在这本书中,作者也对多人游戏编程做了详细的讲解,包括客户端和服务器端的开发,通信协议的设计等,让入门的开发者也能够轻松掌握开发方法。 此外,网络游戏的安全性也非常重要,作者也在书中详细介绍了如何对网络游戏进行安全设计以及避免常见的攻击手段,提高游戏的安全性和稳定性。 总之,网络多人游戏架构与编程 pdf下载对于想要深入了解网络游戏开发的人来说是非常有用的一本参考书,帮助开发者们在游戏开发中更加得心应手。 ### 回答3: 网络多人游戏架构与编程的PDF下载是关于网络游戏相关技术的书籍。对于想要进入游戏开发领域的开发者来说,这是一本非常值得阅读的指南。书中详细介绍了网络游戏的架构和编程技术,并涵盖了游戏开发的整个流程,从概念到最终产品的发布。这本书是由一些资深的游戏开发者撰写,他们提供了丰富的实践经验和建议。 在本书中,开发者们开始了解如何设计和部署服务器架构,以保证游戏流畅运行。他们还学会了如何利用流行的游戏引擎来创建游戏、处理并发和通信,以及保证游戏的可靠性和可扩展性。此外,这本书还介绍了网络多人游戏中的一些重要概念,包括游戏玩法、游戏元素和玩家互动。 总体来说,网络多人游戏架构与编程的PDF下载是一本非常有用的书籍,为新手和有经验的游戏开发者提供了广泛的知识和经验。从技术角度看,它非常详细,并涵盖了游戏开发的各个方面,包括设计、编程和测试。此外,这本书还提供了一些有用的建议,可以帮助开发者在游戏开发的过程中避免常见的问题和陷阱。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值