前言

相信大家都玩过斗地主,规则就不再介绍了。
直接上一张朋友圈看到的残局图:
这道题我刚看到时,曾尝试用手工来破解,每次都以为找到了农民的必胜策略时,最后都发现其实农民跑不掉。由于手工破解无法穷尽所有可能性,所以这道题究竟农民有没有妙手跑掉呢,只能通过代码来帮助我们运算了。
本文将简要讲述怎么通过代码来求解此类问题,在最后会公布残局的最后结果,并开源代码以供大家吐槽。
minimax
代码的核心思想是minimax。minimax可以拆解为两部分,mini和max,分别是最小和最大的意思。
直观的理解是什么呢?就有点像A、B两个人下棋。A现在可以在N个点走棋,假设A在某个点走棋了,使得A的这一步的盘面评估分数最高;但是轮到B下的时候,就一定会朝着让A最不利的方向走,使得A的下一步必然按照B设定的轨迹来,而没法达到A在第一步时估算到这一步的最高盘面评分。
在牌局中是一样的,如果农民的一手牌,让地主无论如何应对都不能赢的话,那么可以说农民有必胜策略;否则,农民必输。
核心逻辑
我们可以用一个函数hand_out来模拟一个人的出牌过程。在现实生活中,一个人想要出牌的话,必然需要知道自己手上的所有牌:me_pokers,也需要知道上一手的出的牌:last_hand。如果我们要用这个函数来模拟两个人的出牌,则还需要知道对手当前的所有牌:enemy_pokers。
这个函数的返回值,是轮到我me_pokers出牌时,是否能够必赢牌。如果能赢则返回真,否则返回假。
def hand_out(me_pokers, enemy_pokers, last_hand)
假设轮到我出牌时,如果我手上的牌都出完了,那么我将立刻知道我赢了;反之如果对手的牌都出完了,而我没有,则我失败了。
if not me_pokers: return True if not enemy_pokers: return False
因为现在轮到我出牌,所以我首先需要知道我现在能出的所有手牌组合。注意:这个组合中,包括过牌(即不出牌)的策略。
all_hands = get_all_hands(me_pokers)
现在我们要对所有可能的手牌组合进行遍历。
首先我需要知道,上一手对方出的牌是什么。
关键点来了,在出完我的牌或选择过牌后,我们需要用一个递归调用来模拟对手下一步的行为。如果对手的下一次出牌不能获胜的话,则我这一次的出牌必胜;否则,对于我的每一个出牌选择,对手都能获胜的话,则我必败。
全部代码如下:
def hand_out(me_pokers, enemy_pokers, last_hand, cache):
if not me_pokers:
# 我全部过牌,直接获胜
return True
if not enemy_pokers:
# 对手全部过牌,我失败
return False
# 获取我当前可以出的所有手牌组合,包括过牌
all_hands = get_all_hands(me_pokers)
# 遍历我的所有出牌组合,进行模拟出牌
for hand in all_hands:
# 如果上一轮对手出了牌,则这一轮我必须要出比对手更大的牌 或者 对手上一轮选择过牌,那么我只需出任意牌,但是不能过牌
if (last_hand and can_comb2_beat_comb1(last_hand, hand)) or (not last_hand and hand['type'] != COMB_TYPE.PASS):
# 模拟对手出牌,如果对手不能取胜,则我必胜
if not hand_out(enemy_pokers, make_hand(me_pokers, hand), hand, cache):
return True
# 如果上一轮对手出了牌,但我这一轮选择过牌
elif last_hand and hand['type'] == COMB_TYPE.PASS:
# 模拟对手出牌,如果对手不能取胜,则我必胜
if not hand_out(enemy_pokers, me_pokers, None, cache):
return True
# 如果之前的所有出牌组合均不能必胜,则我必败
return False
构建
以上核心逻辑理清楚后,构建破解器将变得十分简单。
首先,我们要用数字来表示牌的大小,这里我们用3表示3,11来表示J,12表示Q,依次类推……
其次,我们需要求出一个手牌的所有出牌组合,这里需要get_all_hands函数,具体实现比较繁琐但是很简单,就不在此赘述。
然后,我们还需要一个牌力判断函数can_comb2_beat_comb1(comb1, comb2) ,这个函数用于比较两组手牌的牌力,看是否comb2可以击败comb1。唯一需要注意的一点,在斗地主的规则中,除了炸弹外,其他所有牌力均等,只有牌型一样时才能去比较。
最后,我们需要一个模拟出牌函数make_hand(pokers, hand) ,用于求出在手牌为pokers的情况下打出一手牌hand后,剩下的手牌,实现也非常简单,只需简单的移除掉那些打出的牌即可。
效率
由于一副牌的可能手牌巨大,导致递归的分支数巨大。所以时间开销非常大,为阶乘级O(N!),根据斯特林公式,大约为O(N^N)。
由于可能会有很多重复的牌面出现,导致了很多重复的递归调用。所以加一个缓存能极大提升效率。
即对我方手牌和敌方手牌和上一轮手牌的描述(str(me_pokers)+str(enemy_pokers)+str(last_hand))为键,将求出的结果存进缓存字典中。下一次遇到相同的局面时,即可直接从缓存字典中取出,而无需再次重复计算。时间复杂度优化为指数级O(C^N)。
结果
代码运算出来的结果是,农民没有必胜策略。换言之,只要地主会玩,农民不可能赢。阶级固化已经如斯了么……
开源
代码放于Github: doudizhu_solver,或者大家可以本地下载,MIT协议,随便玩。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。
# python
# 斗地主
# python实现斗地主
# 斗地主残局破解方法
# python模拟斗地主发牌
# python模拟实现斗地主发牌
# python实现斗地主分牌洗牌
# 出牌
# 求出
# 递归
# 出了
# 轮到
# 遍历
# 只需
# 更大
# 我这
# 要用
# 这一轮
# 要知道
# 这道
# 开源
# 支数
# 手上
# 来了
# 不可能
# 本地下载
相关文章:
模具网站制作流程,如何找模具客户?
建站主机数据库如何配置才能提升网站性能?
南宁网站建设制作定制,南宁网站建设可以定制吗?
建站之星多图banner生成与模板自定义指南
七夕网站制作视频,七夕大促活动怎么报名?
如何挑选优质建站一级代理提升网站排名?
如何解决ASP生成WAP建站中文乱码问题?
东莞专业制作网站的公司,东莞大学生网的网址是什么?
建站主机选哪种环境更利于SEO优化?
胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?
实惠建站价格推荐:2025年高性价比自助建站套餐解析
建站之星安装步骤有哪些常见问题?
广州营销型建站服务商推荐:技术优势与SEO优化解析
深圳企业网站制作设计,在深圳如何网上全流程注册公司?
C#怎么创建控制台应用 C# Console App项目创建方法
建站之星如何取消后台验证码生成?
建站之星安装提示数据库无法连接如何解决?
招商网站制作流程,网站招商广告语?
开源网站制作软件,开源网站什么意思?
如何快速生成ASP一键建站模板并优化安全性?
如何做静态网页,sublimetext3.0制作静态网页?
如何通过主机屋免费建站教程十分钟搭建网站?
香港服务器网站卡顿?如何解决网络延迟与负载问题?
完全自定义免费建站平台:主题模板在线生成一站式服务
nginx修改上传文件大小限制的方法
广州网站制作公司哪家好一点,广州欧莱雅百库网络科技有限公司官网?
高端网站建设与定制开发一站式解决方案 中企动力
如何通过虚拟机搭建网站?详细步骤解析
网站制作大概多少钱一个,做一个平台网站大概多少钱?
mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?
小自动建站系统:AI智能生成+拖拽模板,多端适配一键搭建
如何在橙子建站中快速调整背景颜色?
英语简历制作免费网站推荐,如何将简历翻译成英文?
建站ABC备案流程中有哪些关键注意事项?
深圳网站制作平台,深圳市做网站好的公司有哪些?
如何通过建站之星自助学习解决操作问题?
定制建站流程步骤详解:一站式方案设计与开发指南
如何快速选择适合个人网站的云服务器配置?
如何选择适合PHP云建站的开源框架?
如何通过wdcp面板快速创建网站?
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
浙江网站制作公司有哪些,浙江栢塑信息技术有限公司定制网站做的怎么样?
建站之星如何快速解决建站难题?
官网网站制作腾讯审核要多久,联想路由器newifi官网
建站主机与虚拟主机有何区别?如何选择最优方案?
,想在网上投简历,哪几个网站比较好?
网页设计与网站制作内容,怎样注册网站?
建站之星上传入口如何快速找到?
专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?
Android自定义listview布局实现上拉加载下拉刷新功能
*请认真填写需求信息,我们会在24小时内与您取得联系。