Unity3D手机斗地主游戏开发实战(04)_出牌判断大小

简介: 之前我们实现了叫地主、玩家和电脑自动出牌主要功能,但是还有个问题,出牌的时候,没有有效性检查和比较牌力大小。比如说,出牌3,4,5,目前是可以出牌的,然后下家可以出任何牌如3,6,9。 问题1:出牌检查有效性,就是出牌类型判断,像单张、对子、顺子、炸弹等等类型; 问题2:上家出牌后,下家再出牌的时候,要判断当前牌力是否大于上家的牌力; 那本篇我们主要解决以上2个问题。

之前我们实现了叫地主、玩家和电脑自动出牌主要功能,但是还有个问题,出牌的时候,没有有效性检查和比较牌力大小。比如说,出牌3,4,5,目前是可以出牌的,然后下家可以出任何牌如3,6,9。

  • 问题1:出牌检查有效性,就是出牌类型判断,像单张、对子、顺子、炸弹等等类型;
  • 问题2:上家出牌后,下家再出牌的时候,要判断当前牌力是否大于上家的牌力;

那本篇我们主要解决以上2个问题。

卡牌信息类重构

首先,原先的卡牌类,已经实现了单张卡牌牌力的比较,但是有些复杂,我们先对这个比较逻辑进行优化。思路是卡牌的cardIndex就表示在此类型卡牌中的大小权重,所有,在初始化卡牌的过程中,对cardIndex进行特殊的处理:

cardIndex=(卡牌的原始索引+10)%13

比如:普通牌3--cardIndex=(3+10)%13=0,转换后排最小

普通牌J--cardIndex=(11+10)%13=8,转换后排第8张,如此,我们就可以轻松比较cardIndex,来判断单牌的实际大小了。

再看看优化后的代码,是不是比以前简洁很多?更主要的是,为了方便后面的复杂组合牌力的判断。

public class CardInfo : IComparable
{
    public string cardName; //卡牌图片名
    public CardTypes cardType; //牌的类型
    public int cardIndex;      //牌在所在类型的索引3-10,J,Q,K,A,2(0-12)
    public bool isSelected;    //是否选中


    public CardInfo(string cardName)
    {
        this.cardName = cardName;
        var splits = cardName.Split('_');

        switch (splits[1])
        {
            case "1":
                cardType = CardTypes.Hearts;
                cardIndex = (int.Parse(splits[2]) + 10) % 13;
                break;
            case "2":
                cardType = CardTypes.Spades;
                cardIndex = (int.Parse(splits[2]) + 10) % 13;
                break;
            case "3":
                cardType = CardTypes.Diamonds;
                cardIndex = (int.Parse(splits[2]) + 10) % 13;
                break;
            case "4":
                cardType = CardTypes.Clubs;
                cardIndex = (int.Parse(splits[2]) + 10) % 13;
                break;
            case "joker":
                cardType = CardTypes.Joker;
                cardIndex = (int.Parse(splits[2]) + 10) % 13;
                break;
            default:
                throw new Exception(string.Format("卡牌文件名{0}非法!", cardName));
        }
    }

    //卡牌大小比较
    public int CompareTo(object obj)
    {
        CardInfo other = obj as CardInfo;

        if (other == null)
            throw new Exception("比较对象类型非法!");

        //如果当前是大小王
        if (cardType == CardTypes.Joker)
        {
            //对方也是大小王
            if (other.cardType == CardTypes.Joker)
            {
                return cardIndex.CompareTo(other.cardIndex);
            }
            //对方不是大小王
            return 1;
        }
        //如果是一般的牌
        else
        {
            //对方是大小王
            if (other.cardType == CardTypes.Joker)
            {
                return -1;
            }
            //如果对方也是一般的牌
            else
            {
                //计算牌力
                if (cardIndex == other.cardIndex)
                {
                    return -cardType.CompareTo(other.cardType);
                }

                return cardIndex.CompareTo(other.cardIndex);
            }
        }
    }

}
View Code

出牌类型基类

接下来,那怎么判断出牌有效性和出牌的牌力大小判断呢?

我们这样想,每次出牌都是一组牌堆,那我们首先要判断这一组牌堆的类型,比如3带1,单张、炸弹等等;

其次,确定牌堆类型后,我们需要判断这个牌堆是否是合法的,其实就遍历验证是否符合上述的牌堆类型,如果满足,就认为合法,如果所有类型都不满足,则出牌无效,不允许出牌;

再者,怎么判断下家出牌的牌力要大于上家呢,这里我们还得有一个方法,判断相同类型的2个牌堆,牌堆1是否比牌堆2牌力大;

最后,为了实现电脑出牌的AI,还得有自动查找1个跟上家出牌类型一样的牌堆,而且比上家的牌堆大,如果有这种牌,则可以压住上家,否则自动过牌;

有了整体思路,我们这样设计,先定义一个虚基类--出牌类型基类,这里定义了所有需要子类牌堆类型实现的方法

    /// <summary>
    /// 出牌类型基类
    /// </summary>
    public abstract class FollowCardsBase
    {
        /// <summary>
        /// 验证类型
        /// </summary>
        /// <returns></returns>
        public abstract bool Validate(List<CardInfo> cardInfos);
        /// <summary>
        /// 找到最小满足的牌组
        /// </summary>
        /// <returns></returns>
        public abstract List<CardInfo> FindBigger(List<CardInfo> handCardInfos, List<CardInfo> cardInfos);
        /// <summary>
        /// 判断是否牌大过要比较的牌组
        /// </summary>
        /// <param name="handCardInfos"></param>
        /// <param name="cardInfos"></param>
        /// <returns></returns>
        public abstract bool IsBigger(List<CardInfo> handCardInfos, List<CardInfo> cardInfos);
    }
}
View Code

Validate方法:子类实现验证出牌是否满足此类型;

IsBigger方法:给定2个出牌的牌堆,判断此类型牌堆1是否满足牌力大于牌堆2;

FindBigger方法:在给定的手牌中,找出符合类型中满足牌力大于给定牌堆的组合;

这样,我们定义好基类,再利用子类去实现各自的方法,比如对子牌类型的子类,我们判断如果是对子,出牌的时候Validate判断是否也是对子,如果出牌是对子,而且IsBigger

,就允许出牌;当然AI出牌的时候,通过FindBigger,找到满足对子类型,且比给定的对子牌力大的牌组,进行后续出牌操作。

定义各个类型牌型

因为斗地主涉及的牌型有很多,我们可以简单归纳一下。

我这里把单张和顺子作为一种牌组类型来实现,因为考虑顺子其实就是一种单张的特殊情景,只是约束条件是大于等于连续5张的单张牌。所以对子和连对、3带1或3带1的飞机等等,就都可以归为一类,个人感觉实现会简单些。

这里还是以单张和顺子举例:

        /// <summary>
        /// 验证类型
        /// </summary>
        /// <returns></returns>
        public override bool Validate(List<CardInfo> cardInfos)
        {
            cardInfos.Sort();

            if (cardInfos.Count == 1)   //单张
            {
                return true;
            }
            else if (cardInfos.Count >= 5)  //顺子
            {
                //如果最大的牌是王或者2,则不是顺子
                if (cardInfos.Last().cardType == CardTypes.Joker || cardInfos.Last().cardIndex == 12)
                    return false;

                for (int i = 0; i < cardInfos.Count - 2; i++)
                {
                    if (cardInfos[i].cardIndex + 1 != cardInfos[i + 1].cardIndex)
                        return false;
                }
                return true;
            }
            return false;
        }
View Code

按照上一节所述,我们牌组类型子类,首先需要实现Validate方法,来验证是否属于单张或顺子。

  • 如果是一张牌,毫无疑问,是单张
  • 如果是大于等于5张牌,最大的牌不是王或者2,而且是连续的牌,则是顺子
  • 其他情况肯定不是单张或顺子
        /// <summary>
        /// 判断是否牌大过要比较的牌组
        /// </summary>
        /// <param name="handCardInfos"></param>
        /// <param name="cardInfos"></param>
        /// <returns></returns>
        public override bool IsBigger(List<CardInfo> handCardInfos, List<CardInfo> cardInfos)
        {
            cardInfos.Sort();
            handCardInfos.Sort();

            //牌数一样且最小牌比要比较的牌组的最小牌大
            if (handCardInfos.Count == cardInfos.Count && Validate(handCardInfos) && Validate(cardInfos))
            {
                if (handCardInfos[0].CompareTo(cardInfos[0]) > 0)
                    return true;
            }
            return false;
        }
View Code

接着,怎么判断2组牌组的牌力大小呢?

  • 第一步,将2牌组按照从小到大排序
  • 判断2牌组的牌数是否一样
  • 判断2牌组的类型是否都是单张或顺子
  • 如果满足以上2个条件,再判断2牌组的第一张大小
  • 如果牌组1的第一张大于牌组2的第一张,则可以认为牌组1牌力大于牌组2
  • 反之也成立
  • 否则,如果相等,则牌力一样
        /// <summary>
        /// 找到最小满足的牌组
        /// </summary>
        /// <returns></returns>
        public override List<CardInfo> FindBigger(List<CardInfo> handCardInfos, List<CardInfo> cardInfos)
        {
            cardInfos.Sort();
            handCardInfos.Sort();

            if (cardInfos.Count == 1)   //单张
            {
                var cardInfo = handCardInfos.FirstOrDefault(s => s.CompareTo(cardInfos[0]) > 0 && s.cardIndex != cardInfos[0].cardIndex);
                if (cardInfo != null)
                {
                    var result = new List<CardInfo>();
                    result.Add(cardInfo);
                    return result;
                }
                return null;
            }

            else if (cardInfos.Count >= 5)  //顺子
            {
                var count = handCardInfos.Count - cardInfos.Count;

                if (count >= 0)
                {
                    //手牌比牌组多count,则有count + 1可能满足牌组
                    for (int i = 0; i < count + 1; i++)
                    {
                        var mayBiggerCardInfos = handCardInfos.Skip(i).Take(cardInfos.Count).ToList();
                        //是顺子,且最小的牌比要比较的牌组最小牌要大
                        if (Validate(mayBiggerCardInfos) && mayBiggerCardInfos[0].CompareTo(cardInfos[0]) > 0 && mayBiggerCardInfos[0].cardIndex != cardInfos[0].cardIndex)

                        {
                            return mayBiggerCardInfos;
                        }
                    }
                    return null;
                }
                return null;
            }
            return null;
        }
View Code

再来看,我们怎么根据给定的单张或顺子,在手牌中找到对应牌力大于上家的牌组;

这里提供了一个简单的实现方式,找到最小满足的牌组,当然,如果要实现智能的电脑出牌的AI,或者智能提示出牌,肯定要复杂很多,就不在探讨范围内了。

  • 第一步,将2牌组按照从小到大排序
  • 如果上家牌是单张,那简单,从手牌中找到第1张满足牌力大于上家牌的单张即可;
  • 如果上家牌是顺子,判断手牌比上家牌的牌数只差,比如手牌17张,上家牌是5张的顺子,那就是17-5=12,理论上有12+1最多可能满足的牌组组合;
  • 那从第一张手牌开始,遍历12+1次,每次取当前手牌后的5张牌,判断是否比上家牌牌力大,如果不满足,则继续下轮遍历;
  • 如果找到,则返回该牌组,可以认为这牌组牌力大于上家牌,且类型为单张或顺子,牌数跟上家牌一样;
  • 如果遍历完还没找到,认为手牌中没有大于上家牌的牌组,游戏玩家可以跳过次回合;

至此,我们的单张和顺子类型的出牌检验逻辑,大体上就完成了。接下来,我们只需要稍微修改下之前的出牌逻辑~

出牌逻辑调整

这里需要调整2处,一方面,在玩家出牌时,要增加判断,如果玩家选择的牌堆,牌力不够,需要提示玩家,很简单,不在复述;

再来看电脑出牌的逻辑,我们在PlayOthre类中,增加以下这段代码:

if (CardManager._instance.cardManagerState == CardManagerStates.Playing)
            {
                if (Input.GetKeyDown(KeyCode.Q)) //出牌
                {
                    var singleCards = new SingleCards();
                    var cardInfos = singleCards.FindBigger(this.cardInfos, CardManager._instance.currentCardInfos);
                    if (cardInfos != null)
                    {
                        cardInfos.ForEach(s => s.isSelected = true);
                        ForFollow();
                    }
                    else
                    {
                        NotFollow();
                    }
                }
            }
View Code

电脑AI出牌的时候,通过FindBigger,去匹配手牌中有没有满足牌力大于上家牌的牌组类型,如果有,将返回的牌组出出去,如果没有,则自动过牌。

那现在玩家出牌、电脑AI出牌就可以正常处理了,我们来看一下效果:

写在最后

我们的斗地主游戏开发有段时间了,本文就作为阶段性的结束篇~当然,远算不上是一个成品游戏,我只是简单的实现了部分功能,有些代码现在看来还是偷工减料。要完成一个成品游戏,需要大量的精力和时间,条件也不允许,以后有机会的话,可能会对这个项目完善和优化。

这是我第一次写一个系列的东西,写得不好请大家多多包涵~推出这个系列,我的主旨是想和大家分享下Unity3D开发一个斗地主游戏的整体架构和设计模式,虽然是个半成品,但是有些代码我是真的用心去写的,尽量的把我实现的思路和对游戏的理解展现给大家。就像一千个人眼中有一千个哈姆雷特,由于我们对游戏理解的不同,同样的斗地主游戏,不同的开发者,实现的方法可能就是不同的。我能做的,我希望做的,就是把个人的思想剖析给你们,如果你们能理解,或者对你们有所帮助,也是我的幸事。

最后,谢谢各位的一路陪伴。也欢迎大家关注,我们一同学习,共同进步,以后会坚持出新的系列!

资源

项目源码

img_8f0a90f3cbaa0e044fb8bf7b13c4317b.jpe

文章作者:原子蛋
文章出处:https://www.cnblogs.com/lizzie-xhu/
个人网站:https://www.lancel0t.cn/
个人博客:https://blog.lancel0t.cn/
微信公众号:原子蛋Live+
扫一扫左侧的二维码(或者长按识别二维码),关注本人微信公共号,获取更多资源。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

目录
相关文章
|
3月前
|
设计模式 存储 人工智能
深度解析Unity游戏开发:从零构建可扩展与可维护的游戏架构,让你的游戏项目在模块化设计、脚本对象运用及状态模式处理中焕发新生,实现高效迭代与团队协作的完美平衡之路
【9月更文挑战第1天】游戏开发中的架构设计是项目成功的关键。良好的架构能提升开发效率并确保项目的长期可维护性和可扩展性。在使用Unity引擎时,合理的架构尤为重要。本文探讨了如何在Unity中实现可扩展且易维护的游戏架构,包括模块化设计、使用脚本对象管理数据、应用设计模式(如状态模式)及采用MVC/MVVM架构模式。通过这些方法,可以显著提高开发效率和游戏质量。例如,模块化设计将游戏拆分为独立模块。
203 3
|
3月前
|
图形学 开发工具 git
Unity与版本控制:游戏开发团队如何利用Git打造高效协作流程,实现代码管理的最佳实践指南
【8月更文挑战第31天】版本控制在软件开发中至关重要,尤其在Unity游戏开发中,能提升团队协作效率并避免错误。本文介绍如何在Unity项目中应用版本控制的最佳实践,包括选择Git、配置项目以排除不必要的文件、组织项目结构、避免冲突、规范提交信息以及使用分支管理开发流程,从而提高代码质量和团队协作效率。
276 1
|
4月前
|
图形学 机器学习/深度学习 人工智能
颠覆传统游戏开发,解锁未来娱乐新纪元:深度解析如何运用Unity引擎结合机器学习技术,打造具备自我进化能力的智能游戏角色,彻底改变你的游戏体验——从基础设置到高级应用全面指南
【8月更文挑战第31天】本文探讨了如何在Unity中利用机器学习增强游戏智能。作为领先的游戏开发引擎,Unity通过ML-Agents Toolkit等工具支持AI代理的强化学习训练,使游戏角色能自主学习完成任务。文章提供了一个迷宫游戏示例及其C#脚本,展示了环境观察、动作响应及奖励机制的设计,并介绍了如何设置训练流程。此外,还提到了Unity与其他机器学习框架(如TensorFlow和PyTorch)的集成,以实现更复杂的游戏玩法。通过这些技术,游戏的智能化程度得以显著提升,为玩家带来更丰富的体验。
66 1
|
4月前
|
开发者 图形学 Java
揭秘Unity物理引擎核心技术:从刚体动力学到关节连接,全方位教你如何在虚拟世界中重现真实物理现象——含实战代码示例与详细解析
【8月更文挑战第31天】Unity物理引擎对于游戏开发至关重要,它能够模拟真实的物理效果,如刚体运动、碰撞检测及关节连接等。通过Rigidbody和Collider组件,开发者可以轻松实现物体间的互动与碰撞。本文通过具体代码示例介绍了如何使用Unity物理引擎实现物体运动、施加力、使用关节连接以及模拟弹簧效果等功能,帮助开发者提升游戏的真实感与沉浸感。
93 1
|
3月前
|
图形学 C++ C#
Unity插件开发全攻略:从零起步教你用C++扩展游戏功能,解锁Unity新玩法的详细步骤与实战技巧大公开
【8月更文挑战第31天】Unity 是一款功能强大的游戏开发引擎,支持多平台发布并拥有丰富的插件生态系统。本文介绍 Unity 插件开发基础,帮助读者从零开始编写自定义插件以扩展其功能。插件通常用 C++ 编写,通过 Mono C# 运行时调用,需在不同平台上编译。文中详细讲解了开发环境搭建、简单插件编写及在 Unity 中调用的方法,包括创建 C# 封装脚本和处理跨平台问题,助力开发者提升游戏开发效率。
261 0
|
3月前
|
图形学 开发者
透视与正交之外的奇妙视界:深入解析Unity游戏开发中的相机与视角控制艺术,探索打造沉浸式玩家体验的奥秘与技巧
【8月更文挑战第31天】在Unity中,相机不仅是玩家观察游戏世界的窗口,更是塑造氛围和引导注意力的关键工具。通过灵活运用相机系统,开发者能大幅提升游戏的艺术表现力和沉浸感。本文将探讨如何实现多种相机控制,包括第三人称跟随和第一人称视角,并提供实用代码示例。
151 0
|
3月前
|
图形学 开发者
【独家揭秘】Unity游戏开发秘籍:从基础到进阶,掌握材质与纹理的艺术,打造超现实游戏视效的全过程剖析——案例教你如何让每一面墙都会“说话”
【8月更文挑战第31天】Unity 是全球领先的跨平台游戏开发引擎,以其高效性能和丰富的工具集著称,尤其在提升游戏视觉效果方面表现突出。本文通过具体案例分析,介绍如何利用 Unity 中的材质与纹理技术打造逼真且具艺术感的游戏世界。材质定义物体表面属性,如颜色、光滑度等;纹理则用于模拟真实细节。结合使用两者可显著增强场景真实感。以 FPS 游戏为例,通过调整材质参数和编写脚本动态改变属性,可实现自然视觉效果。此外,Unity 还提供了多种高级技术和优化方法供开发者探索。
57 0
|
3月前
|
图形学 开发者 UED
Unity游戏开发必备技巧:深度解析事件系统运用之道,从生命周期回调到自定义事件,打造高效逻辑与流畅交互的全方位指南
【8月更文挑战第31天】在游戏开发中,事件系统是连接游戏逻辑与用户交互的关键。Unity提供了多种机制处理事件,如MonoBehaviour生命周期回调、事件系统组件及自定义事件。本文介绍如何有效利用这些机制,包括创建自定义事件和使用Unity内置事件系统提升游戏体验。通过合理安排代码执行时机,如在Awake、Start等方法中初始化组件,以及使用委托和事件处理复杂逻辑,可以使游戏更加高效且逻辑清晰。掌握这些技巧有助于开发者更好地应对游戏开发挑战。
143 0
|
3月前
|
图形学 开发者 搜索推荐
Unity Asset Store资源大解密:自制与现成素材的优劣对比分析,教你如何巧用海量资产加速游戏开发进度
【8月更文挑战第31天】游戏开发充满挑战,尤其对独立开发者或小团队而言。Unity Asset Store 提供了丰富的资源库,涵盖美术、模板、音频和脚本等,能显著加快开发进度。自制资源虽具个性化,但耗时长且需专业技能;而 Asset Store 的资源经官方审核,质量可靠,可大幅缩短开发周期,使开发者更专注于核心玩法。然而,使用第三方资源需注意版权问题,且可能需调整以适应特定需求。总体而言,合理利用 Asset Store 能显著提升开发效率和项目质量。
83 0
|
4月前
|
开发者 图形学 API
从零起步,深度揭秘:运用Unity引擎及网络编程技术,一步步搭建属于你的实时多人在线对战游戏平台——详尽指南与实战代码解析,带你轻松掌握网络化游戏开发的核心要领与最佳实践路径
【8月更文挑战第31天】构建实时多人对战平台是技术与创意的结合。本文使用成熟的Unity游戏开发引擎,从零开始指导读者搭建简单的实时对战平台。内容涵盖网络架构设计、Unity网络API应用及客户端与服务器通信。首先,创建新项目并选择适合多人游戏的模板,使用推荐的网络传输层。接着,定义基本玩法,如2D多人射击游戏,创建角色预制件并添加Rigidbody2D组件。然后,引入网络身份组件以同步对象状态。通过示例代码展示玩家控制逻辑,包括移动和发射子弹功能。最后,设置服务器端逻辑,处理客户端连接和断开。本文帮助读者掌握构建Unity多人对战平台的核心知识,为进一步开发打下基础。
142 0