热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

房卡麻将分析系列"牌局回放"之数据设计详解及实例

这篇文章主要介绍了房卡麻将分析系列"牌局回放"之数据设计详解及实例的相关资料,需要的朋友可以参考下

房卡麻将分析系列 "牌局回放" 之 数据设计                                                                   

             最近几个月,”房卡“棋牌游戏成为了资本追逐的热点,基于微信的广大用户和社交属性,”房卡”棋牌发展迅速。红孩儿团队因为之前几年有过相关项目的经验积累,鉴于未来广阔的地方棋牌市场和”开发间“机制的发展前景,也开始转向基于”开房间“棋牌游戏的项目开发中。为了更好的与开发者进行交流学习,特开设”房卡麻将游戏分析系列“。

                                                                            红孩儿团队研发的"大赢家"红中麻将

           本套麻将分析基于网络上流传的“网狐”房卡麻将源码做为基础,按照功能模块分为"架设指南",”服务器框架","后台系统","胡牌算法","客户端界面",“防作弊功能”等等细节做一些分析和指导,帮助广大的棋牌游戏开发者迅速掌握“房卡”麻将的研发原理和技巧设计。也希望有兴趣的朋友多多关注。

       第一次开公众号,挑个简单的下手,先来讲一讲房卡麻将中一个重要功能:“牌局回放”,我们都知道,棋牌类游戏注重公平真实不作弊,如果玩家感觉到游戏的过程有作弊,我相信他一定会对这款游戏失去兴趣。但作弊与否,玩家并不容易进行判断。这时候提供一个“牌局回放”功能给玩家进行分析就尤为重要。

       “网狐”等一些长期耕耘在棋牌领域的企业,在这方面都有完整的经验和框架,通过参考,我发现它是通过下面一套流程来完成”牌局回放“功能的。

        首先,在游戏服务器的房间类CTableFrameSink里需要有一个GameRecord结构,这个结构对 玩家信息,手牌以及每一步的动作都可以进行相应的记录:

struct GameRecordPlayer 
{ 
  DWORD dwUserID; 
  std::string kHead; 
  std::string kNickName; 
  std::vector cbCardData; 
  void StreamValue(datastream& kData, bool bSend) 
  { 
    Stream_VALUE(dwUserID); 
    Stream_VALUE(kHead); 
    Stream_VALUE(kNickName); 
    Stream_VECTOR(cbCardData); 
  } 
}; 
 
struct GameRecordOperateResult 
{ 
  enum Type 
  { 
    TYPE_NULL, 
    TYPE_OperateResult, 
    TYPE_SendCard, 
    TYPE_OutCard, 
    TYPE_ChiHu, 
  }; 
 
  GameRecordOperateResult() 
  { 
    cbActiOnType= 0; 
    wOperateUser = 0; 
    wProvideUser = 0; 
    cbOperateCode = 0; 
    cbOperateCard = 0; 
  } 
 
  BYTE    cbActionType; 
  WORD    wOperateUser;            //操作用户 
  WORD    wProvideUser;            //供应用户 
  BYTE    cbOperateCode;           //操作代码 
  BYTE    cbOperateCard;           //操作扑克 
 
  void StreamValue(datastream& kData, bool bSend) 
  { 
    Stream_VALUE(cbActionType); 
    Stream_VALUE(wOperateUser); 
    Stream_VALUE(wProvideUser); 
    Stream_VALUE(cbOperateCode); 
    Stream_VALUE(cbOperateCard); 
  } 
}; 
 
struct GameRecord 
{ 
  std::vector   kPlayers; 
  std::vector kAction; 
   
  void StreamValue(datastream& kData, bool bSend) 
  { 
    StructVecotrMember(GameRecordPlayer, kPlayers); 
    StructVecotrMember(GameRecordOperateResult, kAction); 
  } 
 
  void CleanUp() 
  { 
    kPlayers.clear(); 
    kAction.clear(); 
  } 
}; 

          在datastream.h中,有一套set,get数据流的宏,能够将数据放入到数据流中或从中拿出。

#define Stream_VALUE(Name) \ 
  if(bSend)      \ 
{              \ 
  kData.pushValue(Name);\ 
}\ 
else\ 
{\ 
  kData.popValue(Name);\ 
}\ 

          好了,有了这样一个结构,在游戏开始的时候,我们就可以开始记录本局了。

//游戏开始 
void CTableFrameSink::GameStart() 
{ 
    ... 
    //填充四个玩家的基础信息 
  for (int i = 0; i <4; i++) 
  { 
    GameRecordPlayer  tNewRecordPlayer; 
         
    tagUserInfo *  tpUserInfo = m_pITableFrame->GetTableUserItem(i)->GetUserInfo(); 
    tNewRecordPlayer.dwUserID = tpUserInfo->dwUserID; 
    tNewRecordPlayer.kNickName = tpUserInfo->szNickName; 
     
        //取得手牌信息 
    BYTE cbCardData[MAX_COUNT]; 
    m_GameLogic.SwitchAllToCardData(m_cbCardIndex[i], cbCardData); 
 
    for (int j = 0; j 

        然后我们开始记录操作,分别在玩家出牌,以及玩家应答吃,碰,杠,胡等操作时加入记录。

//用户出牌 
bool CTableFrameSink::OnUserOutCard(WORD wChairID, BYTE cbCardData) 
{ 
     ... 
  //记录动作数据 
  GameRecordOperateResult  tNewRecordOperateResult; 
  tNewRecordOperateResult.cbActiOnType=    GameRecordOperateResult::TYPE_OutCard; 
  tNewRecordOperateResult.cbOperateCard = cbCardData; 
  tNewRecordOperateResult.cbOperateCode = WIK_NULL; 
  tNewRecordOperateResult.wOperateUser = wChairID; 
  tNewRecordOperateResult.wProvideUser = wChairID; 
  m_sGameRecord.kAction.push_back(tNewRecordOperateResult); 
     ... 
} 


//用户操作 
bool CTableFrameSink::OnUserOperateCard(WORD wChairID, BYTE cbOperateCode, BYTE cbOperateCard) 
{ 
     ... 
//记录动作数据 
    GameRecordOperateResult  tNewRecordOperateResult; 
      tNewRecordOperateResult.cbActiOnType= XZDDGameRecordOperateResult::TYPE_OperateResult; 
    tNewRecordOperateResult.cbOperateCard = cbOperateCard; 
    tNewRecordOperateResult.cbOperateCode = cbOperateCode; 
    tNewRecordOperateResult.wOperateUser = wChairID; 
    tNewRecordOperateResult.wProvideUser = m_wProvideUser; 
    m_sGameRecord.kAction.push_back(tNewRecordOperateResult); 
     ... 
} 

          就这样,基本的操作记录也完成了。最后当牌局结束时,我们需要将记录提交到数据库中。

//游戏结束 
bool CTableFrameSink::OnEventGameConclude(WORD wChairID, IServerUserItem * pIServerUserItem, BYTE cbReason) 
{ 
  switch (cbReason) 
  { 
  case GER_NORMAL:    //常规结束 
    { 
         ... 
            //将记录转化为数据流。 
      datastream kDataStream; 
      m_sGameRecord.StreamValue(kDataStream, true);  
            //除去写分等处理,这里最后一个参数即是数据流。 
            m_pITableFrame->WriteTableScore(ScoreInfoArray, CountArray(ScoreInfoArray), kDataStream); 
 
         ... 
        } 
    } 
}  

           在私人场服务器中,会通过WriteTableScore这个函数调用PrivateTableInfo的writeSocre,它将将数的流记录下来。

          并最终在牌局结束时DismissRoom(pTableInfo);发给了数据库。

             数据库最终会通过一个存储过程的执行完成将数据流入库的工作。具体的代码就不再展示了,大家可以参考

CDataBaseEngineSink::OnRequestPrivateGameRecord()。            

           这样一套完整的回放数据流程就结束了。      

           好,今天的分析就到这里,红孩儿欢迎大家下次继续听课哦~

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


推荐阅读
  • 如何实现织梦DedeCms全站伪静态
    本文介绍了如何通过修改织梦DedeCms源代码来实现全站伪静态,以提高管理和SEO效果。全站伪静态可以避免重复URL的问题,同时通过使用mod_rewrite伪静态模块和.htaccess正则表达式,可以更好地适应搜索引擎的需求。文章还提到了一些相关的技术和工具,如Ubuntu、qt编程、tomcat端口、爬虫、php request根目录等。 ... [详细]
  • Linux服务器密码过期策略、登录次数限制、私钥登录等配置方法
    本文介绍了在Linux服务器上进行密码过期策略、登录次数限制、私钥登录等配置的方法。通过修改配置文件中的参数,可以设置密码的有效期、最小间隔时间、最小长度,并在密码过期前进行提示。同时还介绍了如何进行公钥登录和修改默认账户用户名的操作。详细步骤和注意事项可参考本文内容。 ... [详细]
  • 学习SLAM的女生,很酷
    本文介绍了学习SLAM的女生的故事,她们选择SLAM作为研究方向,面临各种学习挑战,但坚持不懈,最终获得成功。文章鼓励未来想走科研道路的女生勇敢追求自己的梦想,同时提到了一位正在英国攻读硕士学位的女生与SLAM结缘的经历。 ... [详细]
  • 云原生边缘计算之KubeEdge简介及功能特点
    本文介绍了云原生边缘计算中的KubeEdge系统,该系统是一个开源系统,用于将容器化应用程序编排功能扩展到Edge的主机。它基于Kubernetes构建,并为网络应用程序提供基础架构支持。同时,KubeEdge具有离线模式、基于Kubernetes的节点、群集、应用程序和设备管理、资源优化等特点。此外,KubeEdge还支持跨平台工作,在私有、公共和混合云中都可以运行。同时,KubeEdge还提供数据管理和数据分析管道引擎的支持。最后,本文还介绍了KubeEdge系统生成证书的方法。 ... [详细]
  • 近年来,大数据成为互联网世界的新宠儿,被列入阿里巴巴、谷歌等公司的战略规划中,也在政府报告中频繁提及。据《大数据人才报告》显示,目前全国大数据人才仅46万,未来3-5年将出现高达150万的人才缺口。根据领英报告,数据剖析人才供应指数最低,且跳槽速度最快。中国商业结合会数据剖析专业委员会统计显示,未来中国基础性数据剖析人才缺口将高达1400万。目前BAT企业中,60%以上的招聘职位都是针对大数据人才的。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • [译]技术公司十年经验的职场生涯回顾
    本文是一位在技术公司工作十年的职场人士对自己职业生涯的总结回顾。她的职业规划与众不同,令人深思又有趣。其中涉及到的内容有机器学习、创新创业以及引用了女性主义者在TED演讲中的部分讲义。文章表达了对职业生涯的愿望和希望,认为人类有能力不断改善自己。 ... [详细]
  • 如何基于ggplot2构建相关系数矩阵热图以及一个友情故事
    本文介绍了如何在rstudio中安装ggplot2,并使用ggplot2构建相关系数矩阵热图。同时,通过一个友情故事,讲述了真爱难觅的故事背后的数据量化和皮尔逊相关系数的概念。故事中的小伙伴们在本科时参加各种考试,其中有些沉迷网络游戏,有些热爱体育,通过他们的故事,展示了不同兴趣和特长对学习和成绩的影响。 ... [详细]
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • 本文介绍了高校天文共享平台的开发过程中的思考和规划。该平台旨在为高校学生提供天象预报、科普知识、观测活动、图片分享等功能。文章分析了项目的技术栈选择、网站前端布局、业务流程、数据库结构等方面,并总结了项目存在的问题,如前后端未分离、代码混乱等。作者表示希望通过记录和规划,能够理清思路,进一步完善该平台。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • 本文介绍了Java的集合及其实现类,包括数据结构、抽象类和具体实现类的关系,详细介绍了List接口及其实现类ArrayList的基本操作和特点。文章通过提供相关参考文档和链接,帮助读者更好地理解和使用Java的集合类。 ... [详细]
author-avatar
青春进行时2502857343
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有