扑克游戏架构及其实现(三)

扑克游戏架构及其实现(三)

技术要点罗列及解析

扑克绘制

定义名词如下:
l         花色suit
扑克的属性之一,所取值为方块、草花、红桃、黑桃之一
l         面值face
扑克的属性之一,所取值为A、2、3等等直到K
l         普通牌
Ace至K的牌,称为 普通牌
l         Joker (大小鬼、大小疯)
除了Ace至K的牌,即Joker(大小鬼、大小疯)
此处笔者参考了 http://www.catch22.net所登载文章《How to use cards.dll》(作者James Brown),笔者原先只看到Part1和Part2,近来有了Part3想来是文章作者推出一个类库,并给出了相关文档。看到与笔者做法有一定雷同之处,不禁有些许欣慰,莫非英雄所见略同尔,呵呵:-)  言归正传
在Cards.dll中,扑克的显示是由一个整数型(integer)数值决定的。设这个值为X,则
u       若X∈[0,52] 则显示为Ace至K的牌,称为 普通牌,其排列顺序是这样的A草花、A方块、A红桃、A黑桃、2草花、2方块、2红桃、2黑桃……本文中,定义此种排序为 面值优先排序 (Face First Sort);若排序为A草花、2草花、3草花……K草花、A方块、2方块、3方块……K方块、A红桃、2红桃、3红桃……K红桃、A黑桃、2黑桃、3黑桃……K黑桃,则称这种排序为 花色优先排序( Suit First Sort )。
u       若X∈[53,68]则显示为一些后背图案,据笔者经验,在不同版本操作系统上,所显示的后背图案不同。XP上就明显要比w2k里漂亮些。
u       若X取其他值,则显示不确定图案,此处要注意,笔者就碰到过取不确定值导致不响应的情况,愿后来者慎之。
故而,扑克对象中宜设置
Øinteger iFace//面值
Øinteger iSuit//花色
又,绘制扑克的关键数值X即为X = iFace + iSuit * 4,在扑克对象中将此值命名为 iFaceFirstValue 另外,某种规则的纸牌游戏还可能用到花色优先的序列,故而又设置 iSuitFirstValue 且该值为iSuitFirstValue=iFace*13+iSuit。对于普通扑克牌来说,由iFace和iSuit值共同决定扑克的牌面,而iFaceFirstValue和iSuitFirstValue亦可用作 排序

对象封装

PokerList
继前文描述的CPoker对象,陆续详解如下:
设置 PokerList类,定义为一组扑克的集合,应当以供至少插入删除检索等函数接口。该类中包含一个指针队列队列(PtrList),大多数功能皆在此队列基础上操作。
Ø         bool CPokerList::SendOut (CPokerList * 函数
Ø         bool CPokerList::IsSentValidate ()函数:
前者函数要求一个参数,即牌需要移动至的目标队列。
后者是判定出牌是否合法的函数,以bool值给出结果。
移动扑克之前,须得选择相关的扑克(调用CPokerWnd::SetPicked(true))。前者调用后者。首先由后者判定出牌是否合法,若合法,则移动,否则退回原PokerList并且设置SetPicked(false)。
Remote与Local
PokerList的继承类 CPlayer有两个继承类分别是 CLocalPlayerCRemotePlayer 其区别在于:
CLocalPlayer为本地操作的玩家,本地玩家操作时,由此出发相关函数从而向远端发送消息,描述所进行的操作,同步数据; CRemotePlayer 则没有这个特性。
在外观的上也可藉此作一些区别设置,比如 CRemotePlayer隐藏牌队列等等。

发牌

在CSDN以往的帖子中,发牌始终是一个难点。笔者的解决方法如下:
用一个一维数组 TotalPoker[N]表示牌盒中的牌,数组的元素个数(设为N)为扑克所用的各色牌总数(同花色计为一张),譬如:若是“拱猪”游戏,没有大小Joker,只有Ace~K的各色牌,则N=52;若是“红五三打一”,有大小Joker,加上Ace~K的各色牌则,共计54张,故N=54。 初始时,数组中的每个元素的值取该花色牌的数目,譬如:“拱猪”需要一副牌,则TotalPoker[n]=1(n∈[0,N-1],n为其间任意整数);“红五三打一”需要两副牌,则TotalPoker[n]=2。
每发一张牌时,随机取i∈[0,N-1](I为整数),若TotalPoker[i]不为零则,分配成功,将TotalPoker[i]减去1否则重复上面的随机过程,并轮换发给各个玩家。以上过程重复N次,则发完所有的牌。
/
// 初始值的设定
for(int i=0;i<52;i++)
       {
              iTotalPokers[i]=1;// 删除队列里面已经有的牌,并把标记数组重新初始化
       }
/
for(i=52;i>0;i--)// 发牌函数里
       {
              while(iTotalPokers[iPokerValueFaceFirst=rand()%52]<1);
              //new a poker and append it to the list
             
              //pPokerWnd=new CPokerWnd(iPokerValueFaceFirst,FALSE);
              pPokerWnd=new CPokerWndPiggie(iPokerValueFaceFirst,FALSE);
              pPokerWnd->Create(NULL,NULL,WS_CHILD | WS_VISIBLE|WS_BORDER|WS_CLIPSIBLINGS ,CRect(0,0,0,0),pParentWnd,ZYY_POKER+i);
              pPokerBox->AddTail(pPokerWnd);
             
              iTotalPokers[iPokerValueFaceFirst]--;
             
       }

出牌次序的控制

以一个CPlayer对象中的 令牌成员 (Token)以及 出牌是否合法判定机制结合即可实现——未得到令牌即可认为出牌不合法。得到令牌并且出牌后,将令牌传递给下一个玩家即可。

出牌的合法性判断

主要通过在继承类中重载CPokerBufferList::IsPokerSentValidate()函数来实现不同游戏中不同的合法规则。需要注意如下规则的考察:
Ø         次序控制
Ø         与首个出牌玩家牌型相同,例如:同数目同花色或同为对子顺子要求等等
Ø         每轮首个出牌的玩家是否有必出的牌,例如:“拱猪”首家必出草花2

拖拉机等牌型的判断(“红五三打一”)

不论花色,仅考以面值。对子、顺子等各种花色的通用判定规则如下:
Ø         将 目标序列(待判断的扑克牌组合)放入一个字符串数组中
Ø         对于不同的 组合,取不同的 标准比较序列,亦放于字符串数组中
Ø         一组牌的面值序列,皆减去最小的面值再加一,得到一个 比较序列
Ø         利用c语言中的字符串相关函数,对比相关的比较序列和目标序列的比较序列,从而判定是否是该种组合。
Ø         特殊情况再作特殊考虑
示例:目标序列为778899,设判断其是否为三连对,该序列的比较序列为112233(=778899-777777+111111),而三连对的标准比较序列为112233。可由strcmp()函数判断得,目标序列就是三连对。
然而,也有特殊情况。譬如:在“红五三打一”中,红桃6作为最大的牌,故而不能参与对子等组合,此时就需要实现判断花色和面值,排除这种情况。

“摔派”的判断(“红五三打一”)

所谓“ 摔牌”,是“红五三打一”等游戏中独特的出牌规则,其作用无异于连续多次出牌,但是必须保证所出牌(组合或单张)是所有玩家手上牌中最大的,目的在于调动对方的牌,减少对方能压牌的可能性,争取主动。
于是如何判断出牌队列中的牌是否最大,极为重要,笔者实现方法如下:
Ø         定义CGame的派生类成员int MaxPokerArray[A][B](通常,A=3,B=4,这是因为),每次出牌前,比较数组中的值和所出的牌值,检查所出的牌是否最大,每次出牌完成后,更新这个数组里面的值。
Ø         利用上述判断方法(“拖拉机”等牌型的判断),将牌序列解析为多个基本的对子或顺子以及单排,分别与MaxPokerArray[3][4]中的值进行比较,若皆为最大,则牌序列为最大的组合,即 拖拉机。

大小比较

Ø         单张牌的比较
单张牌的比较实际上是扑克大小比较的核心——组合的比较最终都归结为单张牌的比较。为每张牌定一个成员变量即 比较值,其取值按照以下规则:
¨         若有主花色规则,对于普通牌,如果是主花色则其iFace值加15(确保大于非主花色的Ace即可)
¨         若规则中Ace或2等某张牌比K大,则再加15
¨         若规则中有 常主牌(比如,“红五三打一”中2为常住),再加15
¨         出一模一样的牌,先出者为大
Ø         对子的比较
比较其中单张牌的 比较值即可
Ø         多张牌组合的比较
假设已经由 合法性判断确认张数相同。则可利用C语言中的函数strcmp()即可实现比较。

规则判定

消息的封装和解析

Ø         封装一个自由操作字符串的类,提供接口:按长度取字符串、按长度添加字符串
Ø         以上述类为基础,以标志位和循环支持流水线消息的封装与解析
参考网站
感谢Mr. James Brown
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值