首页 > 其他分享 >用 Pulsar 开发多人小游戏(一):炸弹人游戏开发需求及难点

用 Pulsar 开发多人小游戏(一):炸弹人游戏开发需求及难点

时间:2023-02-08 13:56:44浏览次数:143  
标签:playerB 游戏 本地 队列 玩家 小游戏 开发 Pulsar 服务端

我之前写过一篇文章 我用消息队列做了个联机游戏 用 Pulsar 这款消息队列实现了一个比较简陋的炸弹人游戏,结果不少读者对这个小游戏都很感兴趣,甚至在 Pulsar 的技术交流群里都遇到了公众号的读者。

所以我决定给这个小游戏开发更多功能,并附上更详尽的文档说明,具体介绍一下这个游戏开发的思路,用到的技术组件以及算法。

没看过之前那篇文章也没关系,这篇文章我将会更具体地介绍这个项目,后续的几篇文章我会逐一介绍每个功能的实现思路。

我已经把比较完善的代码和文档放在了 GitHub 上:

https://github.com/labuladong/play-with-pulsar

下面开始正文,首先要从我小时候特别爱玩的一款游戏说起:

这个游戏叫做 Q 版泡泡堂,应该有不少读者小时候都玩过。游戏里玩家可以操控一个机器人放炸弹,炸开障碍物能够获取随机道具,玩家消灭所有其他机器人则闯关成功,如果被其他机器人消灭,则闯关失败。

这个游戏中其他机器人都是电脑控制的,说实话有些蠢,我玩 Hard 难度一个小时就通关了。所以我在想,是否能够把这类炸弹人游戏做成多人在线的游戏,让几个好朋友联机 PK 呢?

基于这个想法,充分结合现有的技术组件,我计划编写一系列教程:

这个教程的最终产物就是一个多人联机的炸弹人游戏:

游戏功能规划

经典的炸弹人游戏,每个玩家可以移动、放炸弹,炸弹在一段时间后会爆炸,被炸弹炸到的玩家会立即死亡,但允许玩家无限复活。

除了最基本的玩法,我们还有以下需求:

1、需要「房间」的概念,在相同房间里的玩家才能一起对战,不同房间之间不能互相影响。

2、为了提升游戏的操作难度和趣味性,允许玩家推炸弹

4、地图中的障碍物是随机生成的,障碍物分为可摧毁的和不可摧毁的两种类型。考虑到可摧毁的障碍物会被玩家炸掉,我们需要给每个房间定时更新新的地图

5、要有一个房间计分板,显示房间内每个玩家的得分情况。

6、除了当前游戏房间中的分数情况,我们还需要有一个全局计分板,可以对所有玩家在不同房间的总得分进行排名。

7、假设我们会举办重要赛事,需要支持游戏「录制」,以便观看游戏回放

8、最好能够支持 AI 玩家对战,一个人也能玩的很嗨。

多人游戏的难点

我没有专门搞过多人在线游戏的开发,但是简单分析一下,我总结出来以下关键点:

1、多人在线游戏肯定需要有一个后端服务供所有玩家连接,但由于这只是个小游戏,所以希望开发尽可能简单,尽可能少写代码,避免重复造轮子。

2、最重要的,所有玩家的操作必须同步,或者说要保证各个玩家视图的「一致性」

理想情况下,一个玩家做的操作能够通过量子纠缠瞬间同步到所有其他玩家那里,这样各个玩家的视图必然是一致的。

但实际情况是,每个玩家在本地的操作需要通过网络发送到游戏的服务端,然后服务端再通过网络同步给其他玩家。那么这里面多了两次网络通信,随便发生点意外就会破坏每个玩家的视图一致性。

比如就考虑两个玩家 playerAplayerB,他们分别站在 (2, 3)(6, 4),此时每个玩家的本地状态和服务端的状态都是一致的,这很好:

此时 playerA 进行一次移动,先更新本地状态,然后告诉服务端,服务端再通知到 playerB。但是服务端和 playerB 通信时出现了网络抖动导致通信失败,那么就造成了 playerB 的本地状态错误:

playerB 看到 playerA 仍然站在 (2, 3),而实际上 playerA 已经站在了 (3, 3)

此时如果 playerB 攻击 (2, 3),他看见自己攻击了 playerA,但实际上 playerA 并没有受到攻击,这显然是一个非常严重的 bug。

你也许说,如果服务端和 playerB 通信失败,那就重试呗?

实际上也不好搞,因为要保证重试期间 playerB 不能有任何动作,否则 playerB 的本地状态本来就是错的,基于这个错误状态上的所有动作都会让问题更严重。

如何同步玩家

解法其实很简单,我们的后端用一个消息队列就可以解决玩家间同步的问题

1、把所有玩家的操作抽象成一个事件。

2、在服务端有有一个全局一致的事件序列(消息队列)。

3、从一个相同的初始状态开始,执行一系列相同的事件,得到的结果总是相同。

满足了上述条件,所有玩家的本地客户端按顺序消费服务端的全局事件队列,就可以保证每个玩家的本地客户端状态一致了。

综上所述,我们的后端服务就是一个消息队列,客户端本地产生的事件也要先成功发送到消息队列,再从消息队列读取之后才会更新本地状态:

用一段伪码表示可能会更清晰:

// 一个线程负责拉取并显示事件
new Thread(() -> {
    while (true) {
        // 不断从消息队列拉取事件
        Event event = consumer.receive();
        // 然后更新本地状态,显示给玩家
        updateLocalScreen(event);
    }
});

// 一个线程负责生成并发送本地事件
new Thread(() -> {
    while (true) {
        // 本地玩家产生的事件,要发送到消息队列
        Event localEvent = listenLocalKeyboard();
        producer.send(event);
    }
});

这样,所有玩家客户端都以后端消息队列中的事件顺序(全局一致)为准,依次消费这些事件更新本地状态,从而保证了所有客户端的本地状态全局强一致的。

PS:回想一下,我们在玩 MOBA 游戏时,如果由于网络原因短暂卡顿重连,也会出现类似放快速放电影的情况。所以我猜测真实的多人在线游戏可能真的是通过类似消息队列的机制来保证玩家之间同步的。

在下篇文章,我会具体讲讲如何使用 Apache Pulsar 这样一个消息队列实现上面列举的游戏功能。

更多高质量干货文章,请关注我的微信公众号 labuladong 和算法博客 labuladong 的算法秘籍

标签:playerB,游戏,本地,队列,玩家,小游戏,开发,Pulsar,服务端
From: https://www.cnblogs.com/labuladong/p/17101460.html

相关文章

  • iOS SDK开发
    一款好用且设计充分的SDK必须要遵循以下4条基本原则,即:1、SDK安全,稳定2、统一的开发规范3、Library小而精4、不依赖第三方SDK安全,稳定:考虑到SDK是需要嵌入到App......
  • 小程序高级电商前端第3周电商系统分类页面客服商品详情页面开发<三>----为你推荐实现
    为你推荐实现:概述:接着https://www.cnblogs.com/webor2006/p/16645627.html继续往下学习,对于整个首页的功能,还差最后一个木有实现:而这块也是最麻烦的,推荐的商品是需要分......
  • 离线地图的开发笔记(一)
    最近做一个项目,用到离线谷歌地图的开发,而现在谷歌地图在中国的发展不是很完美,导致项目的定项迟迟不能决定。后来制作了一个地图插件mymapw.ocx,完美地解决了Winform窗体里......
  • Pulsar分布式消息系统
    说明ApachePulsar是Apache软件基金会顶级项目,是下一代云原生分布式消息流平台,集消息、存储、轻量化函数式计算为一体。该系统源于Yahoo,最初在Yahoo内部开发和部署......
  • JavaScript 实战开发经验!函数多参数传参技巧
    Tips:必选参数指的是必须要传入实参的参数;可选参数在函数中有默认值,如有传入实参,则取实参的值,如果没有传入实参,则使用默认值。函数的众多参数问题当一个函数既有必选参数,......
  • 开源基于.Net Core 6.0 +DDD模式开发的通用权限系统
    项目简介这是一个基于.NetCore6.0+DDD模式开发的通用权限系统。前端借鉴了诺依的框架,框架选型都是使用了比较最新的技术、前后端分离。技术架构1、跨平台:这是基于.Ne......
  • freeswitch智能语音开发之ASR
    ASR(AutomaticSpeechRecognition)自动语音识别技术是一种将人的语音转换为文本的技术。一、freeswitch如何使用asrfreeswitch提供两个app功能detect_speech和play_and_de......
  • 安装和配置Java开发环境JDK
    我们通常软件开发的操作系统选择Windows,生产环境选择linux或windowsServer。移动开发可能是安卓或IOS和鸿蒙系统等。Windows下一般选择的是64位的操作系统,一般建议CPUI5......
  • 快速上手Java开发工具Eclipse之简易手册
    Eclipse下载,可以下载最新版本,文档是以2020-12R版本为例http://www.eclipse.org/downloads/ 下载Packages即可安装Eclipse 解压安装除了我的电脑----高级---环境变......
  • OpenHarmony开发15 —— 消息队列
    OpenHarmony开发15——消息队列说点别的,这几天没更新真的是被这个消息队列折磨完了,谁知道鬼鸿蒙它不进行任何提示!为什么stackoverflow会不提示啊!!!太折磨了太折磨了......