博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
一个21点游戏的设计与实现
阅读量:6623 次
发布时间:2019-06-25

本文共 10480 字,大约阅读时间需要 34 分钟。

hot3.png

21点游戏 v0.1.0

这是一个纸牌的赌博游戏。

这个游戏的规则:每个玩家分别与庄家比点数,但是又不能超过21点。这是最原始,最简单的规则。但是发展到现代。游戏的规则已经变得复杂了很多。比如双倍下注、分牌、五龙等这样例牌的出现。

具体规则可以在维基百科上搜索“21点”。

架构目标

它们的架构与功能是完全独立的。这意味着我们在构建系统时,应该选择最符合当前需求的架构,并在该架构的框架下实现功能。——《恰如其分的软件架构》

虽然这个21点看似简单,其实不然。当然,如果你只是想实现一个只有最简单21点游戏,而不考虑可扩展性可维护性,你大可使用一个大大的if-else实现。这也算是一种架构。

如果你的目标是:当QQ游戏需要添加21点游戏时,只要在你21点游戏底层库上做些配置就可以直接用了(在写本文时,我真不知道QQ游戏到底有没21点游戏)。 或者某家在线赌博网站想运营一个21点游戏,拿来你的21点游戏底层库,在上面进行配置或扩展,就可以直接用了。

说得再细一些。有些赌场对于Blackjack是3倍赔率,而有些是2倍。有些赌场允许分牌后再分, 有些则不允许。有些赌场只允许在首两张牌时进行加注,而有些赌场允许在任何时候进行加注。

而敝人也希望这个底层库尽可能实现所有的场景。

难点及解决方案

难点在于21点游戏的规则没有标准,每个赌场的规则都可能不同。而我们写出的程序必须应对所有的变化。

解决方案是

  1. 实现21点游戏的领域特定语言,各个赌场只需要修改领域特定语言就可以,而不需要关心具体实现技术
  2. 将游戏中相关稳定的部分(如游戏流程,开牌每人手上都会有两张牌)固化下来,使用java实现。而游戏中变化的部分使用 动态语言实现。

而我采用的是第二种解决方案。

分析与设计

首先要知道21点的游戏流程是不会变的,而变的则是例牌的设置、例牌的赔率,而且同一例牌的规则也可能会不同。

所以,游戏流程可以写死,而例牌方面直接使用动态语言Groovy来实现。

设计中有几个重要的概念需要介绍,这样有利于了解我的设计与架构:

玩家(player):首先庄家有可能是人,也有可能是机器。所以,不论机器或人,在赌桌上所有的人、机器都是玩家。 当赌局里的庄家是随机的时候,意味着所有的玩家都可能做庄家。

玩家代理(playerProxy):有些规则是允许一个玩家下多门注的。所以,我们抽象出玩家代理这个概念。实际掌握手上牌和赌注的是玩家代理。 玩家的所有的操作实际上都是由玩家代理执行。这样就可以解决一个玩家有多门注,及分牌的问题。

庄家代理(dealerProxy):当一个玩家做了庄家,那么他在赌局中实际是通过庄家代理来完成操作。

牌型(cardCategory):牌型指的是手上牌所符合的模式。比如首两张牌组成21点,那么这手牌的牌型就是Blackjack。由于赔率很大程度与牌型及玩 家操作决定,而且每个赌场的规则又不一样。所以,这部分使用Groovy实现。

玩家操作(playerAction):玩家在游戏中只有有限的几个操作:要牌(hit),停牌(stand)等。所以,这部分,直接使用java实现。

银行(bank):这是对赌场中钱的管理人的一个抽象。统一管理所有的钱、判断玩家是否还有足够的钱进行加注等。

赌桌(tables):玩家加入赌桌进行赌博。而赌桌则是所有玩家玩牌的接口。

使用Groovy来实现的好处有:

  1. 实时修改游戏规则(例牌等),不需要重启服务器
  2. 提高程序的可扩展性
  3. 提高游戏的可配置性

为什么不使用XML来配置呢?

鉴于很多人喜欢使用XML做配置文件,我的想法如下:

  1. XML对于游戏规则的表达力不够
  2. 还需要自己写解释器,注意,这个解释器并不是JDOM这样的库,而是指语义解释器。而且,使用java处理XML,你懂的~
  3. 为什么还有多一层解释呢?配置即执行不多好?目前ROR,Gradle,Vagrant也都是配置即执行。
  4. 本人不喜欢XML的哆嗦

当然,如果你希望实现一个使用XML配置的,那也可以。本库的扩展非常好。只需要改WinnerLoserCalculateEngine这个 输赢计算引擎就可以了。

关于例牌

由于例牌是21点中最大的变化,而且各类繁多。所以,我有必要在这里说明一下,哪些例牌是我已经实现并测试了的。

已经实现了的

  1. 保险 保险与玩家下的筹码是独立的。是单独结算的。 所以分牌时,保险金不会分

  2. Blackjack: 开牌后就得到了21点(一张面牌为10的牌,一张A)

  3. 分牌(split) : 玩家首两张牌的点数相同,则可以选择分牌,并须加注。分出的每门的下注金额须与原注相同。 注:有些赌场可分牌后再分,有些则不能

  4. 双倍下注: 发牌后,玩家手上两张牌的点数加起来有11点,就可以进行双倍下注。但是双倍下注后,只能要一张牌。

  5. 过五龙: 玩家或庄家手上的牌超过了5张,但还没有爆牌的情况。

  6. 同花顺: 玩家手上有三张牌:牌色相同,面值分别是6,7,8的情况。

实际上,你可以发挥你的想像来实现你的例牌。比如,当庄家为Blackjack而玩家为五龙,则为打平等等。

未实现的操作

  1. 投降(surrender) 首两张牌可以投降,其他情况不可以投降。投降后,玩家只可以拿回原来赌注的一半。

  2. Ace分牌(AceSplit): 这是分牌的特例。当玩家首两张牌都是A牌时,可以选择分牌,但是分牌后只能得到一张牌。

  3. 退出

分层

在这里,你看不到什么MVC这样的分层方式。是因为我们这个库目前做的事情就是21点最核心的业务。所以MVC这样的分层方式不适合。 目前,你也看不到关于持久化的内容。因为,一持久化是客户的事情,二我目前还没有时间写一个默认的实现。

Blackjack-core 核心包

这是一个核。实现目标是将21点游戏的业务逻辑集中在这个核内。

它可以做什么:

  • 技术上:

    1. 提供方便,简单的接口
    2. 可自定义你的例牌及赔率,并且是实时的,不需要重启服务器
    3. 洗牌算法的可替换
  • 业务上:

    1. 庄家可以是真人,也可以是机器
    2. 支持一个玩家可以下多注
    3. 未限定投注大小,注的限额,我觉得不是21点游戏的核心问题,所以,不实现。
    4. 可以设置保险的赔率

它不可以做什么:

  1. 不可以持久化:玩家数据,金钱数据都存在内存里
  2. 不提供界面
  3. 发牌时的那种一张牌,一张牌的显示的那样的效果。那不应该是核心业务问题。

Blackjack-server 服务器端

目前还没时间做

关于持久化

目前,关于持久化的工作几乎为0。因为我在游戏开发方面没有经验,有一些问题没有想清楚。

  1. 游戏过程的数据是否需要持久? 游戏过程是指记录下玩家的每张出牌,每次的操作。这个过程,我觉得1.完全没有必要;2.性能上的问题。
  2. 游戏中的金钱问题 关于钱的数据当然要持久化了。而且,在设计上要求非常的高。在这方面的我经验也不足。需要从别人系统学习后再设计。

要解决的问题

  1. 如何实现全自动化测试? 也就是不需要任何的mock。玩家自动玩牌。

  2. 如果没有实现自动化测试,那么也就没有测试并发情况也会发生什么。

  3. Groovy脚本配置的带来的安全问题

如何阅读代码

有一个CasinoTest测试类。测试类模拟的是一场简单的赌博。从中可以看出如何使用此核心库。

下面是跑测试时的输出:

12:33:54.568 [main] INFO  c.z.blackjack.core.CasinoTest - 三名玩家分别存入赌场	12:33:54.573 [main] INFO  c.z.blackjack.core.CasinoTest - 甲 存入:400.0	12:33:54.576 [main] INFO  c.z.blackjack.core.CasinoTest - 乙 存入:400.0	12:33:54.577 [main] INFO  c.z.blackjack.core.CasinoTest - 丙 存入:400.0	12:33:54.580 [main] WARN  c.z.blackjack.core.CasinoTest - 三名玩家加入赌桌:1	12:33:54.584 [main] INFO  c.z.blackjack.core.CasinoTest - 选举出庄家为:甲 	12:33:54.586 [main] INFO  c.z.blackjack.core.CasinoTest - 玩家乙 下注:20	12:33:54.586 [main] INFO  c.z.blackjack.core.CasinoTest - 玩家丙 下注:20	12:33:56.154 [main] INFO  c.z.blackjack.core.CasinoTest - *****开始******	12:33:56.154 [main] INFO  c.z.blackjack.core.CasinoTest - 乙 手上的牌,点数:[20] 分别为:[{SPADE-Q}, {SPADE-K}]	12:33:56.154 [main] INFO  c.z.blackjack.core.CasinoTest - 丙 手上的牌,点数:[17] 分别为:[{HEART-J}, {HEART-7}]	12:33:56.236 [main] INFO  c.z.blackjack.core.CasinoTest - 乙 停牌,点数:20,分别为:[{SPADE-Q}, {SPADE-K}]	12:33:56.319 [main] INFO  c.z.blackjack.core.CasinoTest - 丙 停牌,点数:17,分别为:[{HEART-J}, {HEART-7}]	12:33:56.320 [main] INFO  c.z.blackjack.core.CasinoTest - 庄家甲 点数为14不足17点	12:33:56.474 [main] INFO  c.z.blackjack.core.CasinoTest - 庄家要牌,最后点数:[24] 牌为:[{HEART-6}, {HEART-8}, {DIAMOND-K}]	12:33:56.474 [main] INFO  c.z.blackjack.core.CasinoTest - **********所有玩家停牌,开始结算	12:33:56.737 [main] INFO  c.z.blackjack.core.CasinoTest - 乙  下注 20.0 赢家为PLAYER 最后余额:420.0	12:33:56.817 [main] INFO  c.z.blackjack.core.CasinoTest - 丙  下注 20.0 赢家为PLAYER 最后余额:420.0	12:33:56.817 [main] INFO  c.z.blackjack.core.CasinoTest - 庄家甲  最后余额:360.0	12:33:56.818 [main] WARN  c.z.blackjack.core.CasinoTest - *******结束

附录

  1. 保险规则配置

    • 配置文件为类路径下:Insurance.groovy

    • 例子

      //玩家赢时将得到的彩金  winMoney = playerFirstBet * 2
  2. 结算相关配置

    • 实质是配置根据玩家的牌型及庄家的牌来计算输赢,并返回赔率

    • 配置文件为类路径下:Settlement.groovy

    • 例子

      //如果玩家赢,赢多少  winMoney = 0  switch (playerCardCategory) {  //同花顺      case "royal":          winner = PLAYER          winMoney = playerBetSum * 3          break      case "Blackjack":          switch (playerLatestAction) {              case stand:                  if (isDealerBlackjack) {                      winner = PUSH                  } else if (!isDealerBlackjack) {                      winner = PLAYER                      winMoney = playerBetSum * 3                  }              default:                  winner = PLAYER                  winMoney = playerBetSum * 3                  break          }          break      case "fiveCard":          winner = PLAYER          winMoney = playerBetSum * 3          break      case "bust":          winner = DEALER          break      default:          //庄家爆牌          if (dealerSumPoint > 21) {              if(playerSumPoint <= 21){                  winner = PLAYER                  winMoney = playerBetSum * 1;                  break              }              if (playerSumPoint > 21) {                  winner = PUSH                  winMoney = 0                  break;              }          }else{              if (playerSumPoint > dealerSumPoint) {                  winner = PLAYER                  winMoney = playerBetSum * 1              } else if (playerSumPoint < dealerSumPoint) {                  winner = DEALER              } else {                  winner = PUSH              }          }          break  }
  3. 牌型配置

    • 实质是根据玩家手上的牌,计算出手上牌所属的牌型

    • 配置文件为类路径下:Special.groovy

    • 例子

      switch (_headCard.cardCount) {      case 2:          if (_headCard.sumPoint == 21) { // 首两张牌点数为21点时是Blackjack              name = "Blackjack"              actions = [_report]          } else if (_headCard.sumPoint == 11) {  // 首两张下牌点数等于11点时可以双倍下注              name = "doubleDown"              actions = [_hit, _stand, _doubleDown]          } else if (_headCard.isFirstTwoCardPointEquals()) { //首两张牌的面值相同,可进行分牌              //设置分牌后不可以再分。              if (_headCard.isSplit()) {                  return              }              name = "split"              actions = [_hit, _stand, _split]          }          break      case 3:          // 同花顺          if (_headCard.containsAllCardFace(6, 7, 8) && _headCard.isCardFaceIdentical()) {              name = "royal"              actions = [_report]          }          break      case 5:          if (_headCard.sumPoint <= 21){              name = "fiveCard"              actions = [_report]          }          break  }  // 爆牌  if (_headCard.sumPoint > 21) {      name = "bust"      actions = [_stop]  }
  4. 在配置脚本中可以使用的变量 :

    binding.setVariable("PLAYER", Winner.PLAYER);  binding.setVariable("DEALER", Winner.DEALER);  binding.setVariable("PUSH", Winner.PUSH);  binding.setVariable("playerBetAmounts", playerProxy.getBet().getBetAmounts());  binding.setVariable("playerFirstBet", playerProxy.getBet().getBetAmounts().get(0));  binding.setVariable("playerBetSum", playerProxy.getBet().getBetSum());  binding.setVariable("playerLatestAction", playerProxy.getLatestActionName());  binding.setVariable("isPlayerBlackjack", playerProxy.isBlackJack());  binding.setVariable("playerCardCategory", playerProxy.getCardCategoryName());  binding.setVariable("playerSumPoint", playerProxy.getSumPoint());  binding.setVariable("playerCardCount", playerProxy.getHeadCard().getCardCount());  binding.setVariable("dealerCardCategory", dealerProxy.getCardCategoryName());  binding.setVariable("dealerSumPoint", dealerProxy.getSumPoint());  binding.setVariable("dealerCardCount", dealerProxy.getHeadCard().getCardCount());  binding.setVariable("isDealerBlackjack", dealerProxy.isBlackJack());  binding.setVariable(StandAction._name, StandAction._name);  binding.setVariable(HitAction._name, HitAction._name);  binding.setVariable(ReportAction._name, ReportAction._name);  binding.setVariable(DoubleDownAction._name, DoubleDownAction._name);  binding.setVariable(SplitAction._name, SplitAction._name);  binding.setVariable(StopAction._name, StopAction._name);  binding.setVariable(SurrenderAction._name, SurrenderAction._name);

这是虚拟玩游戏时的打印。运行CasinoTest.java就可以了

12:35:31.278 [main] INFO c.z.blackjack.core.CasinoTest - 三名玩家分别存入赌场 12:35:31.283 [main] INFO c.z.blackjack.core.CasinoTest - 甲 存入:400.0 12:35:31.285 [main] INFO c.z.blackjack.core.CasinoTest - 乙 存入:400.0 12:35:31.286 [main] INFO c.z.blackjack.core.CasinoTest - 丙 存入:400.0 12:35:31.289 [main] WARN c.z.blackjack.core.CasinoTest - 三名玩家加入赌桌:1 12:35:31.292 [main] INFO c.z.blackjack.core.CasinoTest - 选举出庄家为:丙 12:35:31.293 [main] INFO c.z.blackjack.core.CasinoTest - 玩家甲 下注:20 12:35:31.293 [main] INFO c.z.blackjack.core.CasinoTest - 玩家乙 下注:20 12:35:32.761 [main] INFO c.z.blackjack.core.CasinoTest - 开始* 12:35:32.762 [main] INFO c.z.blackjack.core.CasinoTest - 甲 手上的牌,点数:[11] 分别为:[{HEART-2}, {DIAMOND-9}] 12:35:32.762 [main] INFO c.z.blackjack.core.CasinoTest - 乙 手上的牌,点数:[15] 分别为:[{DIAMOND-Q}, {SPADE-5}] 12:35:33.092 [main] INFO c.z.blackjack.core.CasinoTest - 甲 双倍下注,最后点数:[20],牌为:[{HEART-2}, {DIAMOND-9}, {DIAMOND-9}] 12:35:33.233 [main] INFO c.z.blackjack.core.CasinoTest - 乙 停牌,点数:15,分别为:[{DIAMOND-Q}, {SPADE-5}] 12:35:33.233 [main] INFO c.z.blackjack.core.CasinoTest - 庄家丙 点数为11不足17点 12:35:33.322 [main] INFO c.z.blackjack.core.CasinoTest - 庄家要牌,最后点数:[20] 牌为:[{DIAMOND-8}, {SPADE-3}, {DIAMOND-9}] 12:35:33.322 [main] INFO c.z.blackjack.core.CasinoTest - **********所有玩家停牌,开始结算 12:35:33.536 [main] INFO c.z.blackjack.core.CasinoTest - 甲 下注 40.0 赢家为PUSH 最后余额:400.0 12:35:33.593 [main] INFO c.z.blackjack.core.CasinoTest - 乙 下注 20.0 赢家为DEALER 最后余额:380.0 12:35:33.594 [main] INFO c.z.blackjack.core.CasinoTest - 庄家丙 最后余额:420.0 12:35:33.594 [main] WARN c.z.blackjack.core.CasinoTest - *******结束

转载于:https://my.oschina.net/zjzhai/blog/203842

你可能感兴趣的文章
zabbix企业应用之分布式监控proxy
查看>>
【Android游戏开发二十六】追加简述SurfaceView 与 GLSurfaceView效率!
查看>>
【OpenCV学习】运动检测实例
查看>>
Java中字符流与字节流的区别
查看>>
winform下的一个分页控件总结
查看>>
arcgis engine 获取高亮Feature、element
查看>>
Linux--U盘安装Ubuntu12.04
查看>>
Linux 小知识翻译 - 「packet」(网络数据包)
查看>>
Caliburn.Micro学习笔记(三)----事件聚合IEventAggregator和 Ihandle<T>
查看>>
NeHe OpenGL教程 第二十五课:变形
查看>>
你真的了解事务吗?
查看>>
心得:对AMF3的误解
查看>>
将你的Asp.NET应用程序嵌入到SharePoint 读后感[转]
查看>>
有关项目上潜在需要的移动端GIS系统源码整理,待后续更新
查看>>
[QQ游戏]五子棋WG 1.0
查看>>
VBA中的错误处理
查看>>
scrapy 的 selector 练习
查看>>
Android DiskLruCache完全解析,硬盘缓存的最佳方案
查看>>
实现开发板与ubuntu的共享--根文件系统NFS--Samba共享【sky原创】
查看>>
移动开发iOS&Android对比学习--异步处理
查看>>