五邑隐侠,本名关健昌,12年游戏生涯。 本教程以Go语言为例。
 
网络游戏程序分为客户端和服务端。客户端负责图形渲染、交互和一些简单校验处理,服务端负责业务逻辑处理、数据存储。
我们开发一个游戏demo,服务端程序可以是一个单线程的服务进程。它包含网络通信、业务逻辑处理、数据存储。服务端打开网络端口监听,客户端通过网络连接到服务端,服务端接入连接。客户端发包给服务端,服务端接收到包后进行解析,调用对应的处理程序进行处理,处理程序处理成功后,修改数据并保存下来,再把响应包封包发送给客户端。
简单的数据存储可以保存在文件里。当用户量逐渐增加,数据存储的性能、完整和安全要求逐渐增大,数据存储改为存储到数据库。这样服务端就分为服务进程、数据库进程两个进程。服务进程跟数据库进程通信,进行数据读写。由于游戏的实时要求比较高,如果每个请求都要读写数据库,当大量用户同时访问时,数据库读写成了服务性能的瓶颈。因此,一般游戏都做缓存,简单的缓存可以缓存在服务进程的内存里,业务处理直接操作缓存数据,每隔一段时间(例如10分钟),用个独立线程把被修改的缓存数据保存到数据库。
这样会有一个问题,就是一旦服务进程宕机,在保存间隔时间里的游戏数据就会丢失掉。特别是游戏服务进程有更新上线时,稳定性还没有被线上并发验证,宕机的几率会增加,数据丢失的风险也会增加。为了减轻风险,可以考虑把数据缓存跟服务进程分离。使用共享内存,或者使用第三方的专业缓存程序(memcached、redis)作缓存。这样数据丢失的风险就大大降低了。
随着游戏业务开发,游戏业务不再是简单的请求、返回,它还有一系列推送需求,如聊天、邮件系统,这类需求可能是全员推送,这会占用服务进程的CPU时间,导致请求响应变慢。而且这种推送时效性要求没有游戏其他业务高,因此可以把这种广播类型的推送抽离出来放到一个单独的广播进程,服务进程通过向广播进程发送广播消息到广播队列,广播进程从广播队列获取消息进行广播。这样又产生了新的问题,客户端需要连接到两个服务端的进程。为了解决这个问题,可以增加网关进程,客户端只需要连接到网关进程,由网关进程代替客户端将请求发送到服务进程进行处理,服务进程、广播进程的响应/推送进入网关,由网关确定转发给哪个客户端连接。
这样服务端就变成了5个独立进程的进程组,接下来对数据库和数据缓存做下技术选型,可以选择业界用得比较多的mysql数据库、redis缓存。利用redis的list结构做消息队列建立起服务进程和广播进程的通信。
这些进程都部署在一台机器上,所以目前服务端的处理能力受限于一台机器的硬件性能。当然我们也可以将这5个进程分散到多台机器上,如果玩家人数再增加,可以把网关进程、服务进程和广播进程集群,redis、mysql改成分布式缓存和分布式数据库,这样也能扩容。更通用的做法是对游戏分服,每个分服的数据互相隔离。这是目前市面上大多数游戏的做法,滚服运营也已经是比较成熟的运营手段。由这5个进程组成的进程组,1个分服对应一个进程组,一个进程组部署在同一台机器上。这样通过分服就可以横向支持更多的玩家并发访问。
这样又出现了新的问题,客户端怎么知道该连接哪个分服。而且现在的游戏,同一个玩家可以在不同的分服进行游戏,以达到好友同玩一个服。一般的,在玩家进入分服前,让玩家先进行全局登录,然后根据游戏类型,要么下发分服列表由玩家自己选择分服,要么给玩家指定分服(例如类似COC的所谓不分服游戏,这个分服可以理解为一个节点)。增加一个全局的账号进程,负责游戏的全局登录、下发分服信息。账号进程还负责支付回调,进行订单确认和发货。
随着游戏运营推广,分服数量越来越多,游戏迭代也需要对服务进行维护,靠人工维护分服列表显得越来越笨拙。应该建立一个自动化的机制,当开新分服,或者分服进入维护的时候,自动更新分服列表。在这里可以通过服务注册发现的机制来实现自动化。利用一个全局的redis作为注册中心,所有分服的信息通过服务编号作为字段,都保存在redis的一个hash结构里,key为gamesrv。当开新分服,或者切换分服状态时,分服的服务进程向redis更新自己所在分服的信息,然后通过订阅发布机制,发布gamesrv通道的消息,参数为更新的服务编号。账号进程在玩家全局登录时,只需要把redis里key为gamesrv的hash返回给客户端就可以。
还有一种情况是,分服宕机了,这个时候需要修改分服状态为维护中,避免玩家进入这个分服。而且redis里分服的信息也应该落地固化,这样就算redis重启了也不丢失。这里可以利用redis的固化机制保存数据。增加一个管理进程,redis数据有变更就调用一次redis的bgsave命令。当然也可以搭建一个注册进程,提供类似redis的订阅发布机制,以及zookeeper的连接监听、树形数据保存、落地固化。管理进程与所有分服的服务进程保持连接,当分服宕机时,管理进程知道对应分服的连接断开,则修改redis里对应分服的信息状态为维护中,这样管理进程也起到监控作用。管理进程除了参与服务的注册发现,还提供对服务和玩家进行管理和操作的服务,管理进程与账号进程保持连接,账号进程支付确认后可以通过管理进程通知具体的分服进行虚拟道具发货。
 
账号进程处理全局登录,那就要有一个账号的数据库用来保存玩家的账号信息,分服可以通过管理进程把分服创角、角色更新信息保存到账号数据库,以便玩家选择分服时知道各分服角色的信息。玩家进行全局登录是短暂的一次请求,所以用http短连接来增加访问量。账号进程支持集群,这样需要有对外统一的访问地址。管理进程也提供一些http请求给管理后台网页对服务和玩家进行管理和操作。增加一个http网关提供统一的地址访问,可以用业界普遍使用的nginx作为http网关。
对于轻中度游戏,游戏的通信量不会很多,没必要每个分服都有一个长连接socket网关。假设一个分服同时连接服务器的客户端有5k,一台机器的socket网关能支持5w个玩家。这样可以把网关独立出来,一台机器部署两个socket网关,每个socket网关都可以进入任意一个分服。这样子可以通过加减机器横向伸缩socket网关,把网络流量成本集中在几台网关机器上降低成本。这样子账号进程就需要知道网关列表和负载情况,以通过负载均衡给玩家返回合适的连接网关。每个网关也要监听各个分服的状态变化,确定是否连接对应的分服的服务进程、广播进程。因此网关需要参与服务的注册发现。
这样一个服务端的架构就基本出来了。从图中可以看出,最大单点风险是管理进程。可以通过守护进程让管理进程崩溃自动重启(守护进程作为父进程创建、等待作为子进程的管理进程退出后,如果没有维护标记则重新创建子进程),提高整个服务的健壮性。
产品上线后还要有数据统计后台,数据来源可以通过账号进程、服务进程运行时生成日志到日志文件,由脚本分析日志上报给统计数据库。统计后台可以作为一个单独的项目去做建表、上报、生成报表。
服务端架构就介绍到这里,接下来聊一下网络通信。

go语言游戏服务端开发(一)——架构的更多相关文章

  1. go语言游戏服务端开发(三)——服务机制

    五邑隐侠,本名关健昌,12年游戏生涯. 本教程以Go语言为例.   P2P网络为服务进程间.服务进程与客户端间通信提供了便利,在这个基础上可以搭建服务. 在服务层,通信包可以通过定义协议号来确定该包怎 ...

  2. go语言游戏服务端开发(二)——网络通信

    一.网络层 网络游戏客户端除了全局登录使用http请求外,一般通过socket长连接与服务端保持连接.go语言的net包提供网络socket长连接相关操作. 对于服务端,一般经历 Listen.Acc ...

  3. go语言游戏服务端开发(四)——RPC机制

    五邑隐侠,本名关健昌,12年游戏生涯. 本教程以Go语言为例. RPC指远程方法调用,游戏里引入RPC目的是降低跨进程交互的复杂度. 游戏业务设计为多go routine,一个玩家一个go routi ...

  4. 转: 基于netty+ protobuf +spring + hibernate + jgroups开发的游戏服务端

    from: http://ybak.iteye.com/blog/1853335 基于netty+ protobuf +spring + hibernate + jgroups开发的游戏服务端 游戏服 ...

  5. 从架构师视角看是否该用Kotlin做服务端开发?

    前言 自从Oracle收购Sun之后,对Java收费或加强控制的尝试从未间断,谷歌与Oracle围绕Java API的官司也跌宕起伏.虽然Oracle只是针对Oracle JDK8的升级收费,并释放了 ...

  6. 为什么多数游戏服务端是用 C++ 来写

    早年开发游戏必须用C++,这没得说,2000-2004年,java还没有nio,其他动态语言不抗重负,只能C/C++能开发出完整可用的游戏服务端.直到2005年,韩国的游戏很多都还是纯C++写服务端, ...

  7. 俯瞰 Java 服务端开发

    原文首发于 github ,欢迎 star . Java 服务端开发是一个非常宽广的领域,要概括其全貌,即使是几本书也讲不完,该文将会提到许多的技术及工具,但不会深入去讲解,旨在以一个俯瞰的视角去探寻 ...

  8. 游戏服务端中使用Servlet和Java注解的一个好设计

    SNS类游戏基本都是使用HTTP短连接,用Java来开发服务端时能够使用Servlet+Tomcat非常轻松的架构起服务端来.在这里介绍一种使用Servlet比較好的一种设计,我也见过非常多基于HTT ...

  9. Day01_搭建环境&CMS服务端开发

    学成在线 第1天 讲义-项目概述 CMS接口开发 1 项目的功能构架 1.1 项目背景 受互联网+概念的催化,当今中国在线教育市场的发展可谓是百花齐放.如火如荼. 按照市场领域细分为:学前教育.K12 ...

随机推荐

  1. CentOS Linux Cockpit 管理工具使用

    1.安装 # yum install  cockpit 2.启动服务 # systemctl start cockpit.socket 3.设置开机启动 # systemctl enable  coc ...

  2. 守护线程_daemon

    守护线程_daemon 线程分为用户线程和守护线程 虚拟机必须确保用户线程(main)执行完毕 虚拟机不用等待守护线程(gc)执行完毕 如:后台记录操作日志,监控内存,垃圾回收等等 测试案例: pac ...

  3. spring-session-data-redis反序列化问题

    springCloud项目,采用springSession,用户模块同时引入了spring-cloud-starter-security,在其他模块request.getSession()的时候抛了以 ...

  4. SpringBoot开发十-开发登录,退出功能

    需求介绍-开发登录,退出功能 访问登录页面:点击头部区域的链接打开登录页面 登录: 验证账号,密码,验证码 成功时生成登录凭证发放给客户端,失败时跳转回登录页面 退出: 将登录状态修改为失效的状态 跳 ...

  5. 捉虫日记 | MySQL 5.7.20 try_acquire_lock_impl 异常导致mysql crash

    背景 近期线上MySQL 5.7.20集群不定期(多则三周,短则一两天)出现主库mysql crash.触发主从切换问题,堆栈信息如下: 从堆栈信息可以明显看出,在调用 try_acquire_loc ...

  6. 教你使用ApiPost中的全局参数和目录参数

    前面的示例中,我们都是在单一接口中填入不同的请求header.query.body参数.但在实际项目中,对于一批接口,往往具有相同的请求参数.此时,我们可以利用全局参数或者目录参数实现. 例如:常见的 ...

  7. [ES6深度解析]13:let const

    当Brendan Eich在1995年设计了JavaScript的第一个版本时,他犯了很多错误,包括从那时起就成为该语言一部分的一些错误,比如Date对象和当你不小心将它们相乘时对象会自动转换为NaN ...

  8. 信号量-Semaphore、SemaphoreSlim

    核心类:Semaphore,通过int数值来控制线程个数. * 通过观察构造函数 public Semaphore(int initialCount, int maximumCount);: * in ...

  9. C语言格式化输出语句

    %d:带符号十进制整数 : %c:单个字符: %s:字符串: %f:6位小数:float; %.2f表示小数点后精确到两位 %lf:6位小数:double;

  10. TFRecord读写简介+Demo 基于Ubuntu18.04+Tensorflow1.12 无WARNING

    简介 TFRecord是TensorFlow官方推荐使用的数据格式化存储工具. 它规范了数据的读写方式. 只要生成一次TFRecord,之后的数据读取和加工处理的效率都会得到提高. 将图片转换成TFR ...