Python 案例分析|21 点扑克牌游戏

简介: 本案例通过一个 21 点扑克牌游戏的设计和实现,帮助大家了解使用 Python 数据类型、控制流程和输入输出。

image.png


本案例通过一个 21 点扑克牌游戏的设计和实现,帮助大家了解使用 Python 数据类型、控制流程和输入输出。
01、21 点扑克牌游戏概述
21 点又名黑杰克(Blackjack),是一种流行的扑克游戏。该游戏由两到六个人玩,使用除大小王之外的 52 张牌,游戏者的目标是使手中的牌的点数之和不超过 21 点且尽量大。

一手扑克牌的点数的计算规则如下:2 至 9 牌,按其原点数计算;10、J、Q、K 牌都算作 10 点(一般记作 T,即 Ten);A 牌(Ace)既可以算作 1 点也可以算作 11 点,由玩家自己决定(当玩家停牌时,点数一律视为最大而尽量不爆,如 A+K 为 21,A+5+8 为 14 而不是 24)。

02、21 点扑克牌游戏设计思路
本节按下列规则模拟 21 点扑克牌游戏:计算机人工智能 AI 作为庄家(House),用户作为玩家(Player)。

游戏开始时,庄家从洗好的一副牌中发牌:第 1 张牌发给玩家,第 2 张牌发给庄家,第 3 张牌发给玩家,第 4 张牌发给庄家。

然后,询问玩家是否需要继续“拿牌”,通过一次或多次“拿牌”,玩家尝试使手中扑克牌的点数和接近 21。如果玩家手中扑克牌的点数之和超过 21,则玩家输牌。

当玩家决定“停牌”(即,不再“拿牌”),则轮到庄家使用下列规则(“庄家规则”)“拿牌”:如果庄家手中的最佳点数之和小于 17,则必须“拿牌”;如果点数之和大于或等于 17,则“停牌”。如果庄家的点数之和超过 21,则玩家获胜。

最后,比较玩家和庄家的点数。如果玩家的点数大,则获胜。如果玩家的点数小,则输牌。如果点数相同,则平局。但玩家和庄家的牌值都是 21 点,此时拥有 blackjack(一张 Ace 和一张点数为 10 的牌)方获胜。

程序的流程如下:

(1)初始化一副洗好的扑克牌(调用函数 get_shuffled_deck()),初始化庄家和玩家手中的牌为空;

(2)依次给玩家和庄家各发两张牌(调用函数 deal_card());

(3)玩家拿牌:询问玩家是否继续拿牌,如果是,继续给玩家发牌(调用函数 deal_card()),并计算玩家牌点,如果大于 21 点,输出“玩家输牌!”信息,并返回。

(4)庄家拿牌:庄家(计算机人工智能 AI)按“庄家规则”确定是否拿牌,如果是,继续给庄家发牌(调用函数 deal_card()),并计算庄家牌点,如果大于 21 点,输出“玩家赢牌!”信息,并返回。

(5)分别计算庄家和玩家的点数,比较点数大小,输出输赢结果信息。

03、21 点扑克牌游戏实现
【例 1】猜 21 点扑克牌游戏示例程序(blackjack.py)。

import random
def get_shuffled_deck():
    """初始化包括52张扑克牌的列表,并混排后返回,表示一副洗好的扑克牌"""

    # 花色suits和序号
    suits = {'♣', '♠', '♢', '♡'}
    ranks = {'2','3','4','5','6','7','8','9','10','J','Q','K','A'}
    deck = []
    # 创建一副52张的扑克牌
    for suit in suits:
        for rank in ranks:
            deck.append(rank+' '+suit)
    random.shuffle(deck) #混排,即洗牌
    return deck
def deal_card(deck, participant):
    """发一张牌给参与者participant"""
    card = deck.pop()
    participant.append(card)
    return card
def compute_total(hand):
    """计算并返回一手牌的点数和"""
    values = {'2':2, '3':3, '4':4, '5':5, '6':6, '7':7, '8':8,
              '9':9, '1':10, 'J':10, 'Q':10, 'K':10, 'A':11}
    result = 0  #初始化点数和为0
    numAces = 0 #A的个数
    # 计算点数和A的个数
    for card in hand:
        result += values[card[0]]
        if card[0] == 'A':
            numAces += 1
    # 如果点数和>21,则尝试把A当做1来计算
    #(即减去10),多个A循环减去10,直到点数<=21
    while result > 21 and numAces > 0:
        result -= 10
        numAces -= 1
    return result
def blackjack():
    """21点扑克牌游戏,计算机人工智能AI为庄家,用户为玩家"""
    #初始化一副洗好的扑克牌,初始化庄家和玩家手中的牌为空
    deck = get_shuffled_deck()
    house = [] # 庄家的牌
    player = [] # 玩家的牌
    #依次给玩家和庄家各发两张牌
    for i in range(2): # 开始发两轮牌
        deal_card(deck, player) # 给玩家发一张牌
        deal_card(deck, house) # 给庄家发一张牌
    # 打印一手牌
    print('庄家的牌:', house)
    print('玩家的牌:', player)
    # 询问玩家是否继续拿牌,如果是,继续给玩家发牌
    answer = input('是否继续拿牌(y/n,缺省为y): ')
    while answer in ('', 'y', 'Y'):
        card = deal_card(deck, player)
        print('玩家拿到的牌为:{0}, {1}'.format(card, player))
        #计算牌点
        if compute_total(player) > 21: # 如果大于21点
            print('爆掉 ... 玩家输牌!')
            return
        answer = input('是否继续拿牌(y/n,缺省为y): ') #继续询问是否拿牌
    # 庄家(计算机人工智能)按“庄家规则”确定是否拿牌
    while compute_total(house) < 17:
        card = deal_card(deck, house)
        print('庄家拿到的牌为:{0}, {1}'.format(card, house))
        #计算牌点
        if compute_total(house) > 21: # 如果大于21点
            print('爆掉 ... 玩家赢牌!')
            return

    #分别计算庄家和玩家的点数,比较点数大小,输出输赢结果信息
    houseTotal, playerTotal = compute_total(house), compute_total(player)
    if houseTotal > playerTotal:
        print('庄家赢牌!')
    elif houseTotal < playerTotal:
        print('玩家赢牌!')
    elif houseTotal == 21 and 2 == len(house) < len(player):
        print('You loose.') # 拥有blackjack的庄家赢牌
    elif playerTotal == 21 and 2 == len(player) < len(house):
        print('庄家赢牌!') # 拥有blackjack的玩家赢牌
    else:
        print('平局!')

if __name__ == '__main__':
    blackjack()

运行结果示例 1。

庄家的牌: ['8 ♠', 'J ♢']

玩家的牌: ['3 ♢', '10 ♠']

是否继续拿牌(y/n,缺省为 y):

玩家拿到的牌为:9 ♡, ['3 ♢', '10 ♠', '9 ♡']

爆掉 ... 玩家输牌!

运行结果示例 2。

庄家的牌: ['K ♡', '2 ♢']

玩家的牌: ['7 ♣', 'Q ♠']

是否继续拿牌(y/n,缺省为 y): n

庄家拿到的牌为:K ♠, ['K ♡', '2 ♢', 'K ♠']

爆掉 ... 玩家赢牌!

目录
相关文章
|
3天前
|
人工智能 数据挖掘 大数据
538个代码示例!麻省理工教授的Python程序设计+人工智能案例实践
Python简单易学,且提供了丰富的第三方库,可以用较少的代码完成较多的工作,使开发者能够专注于如何解决问题而只花较少的时间去考虑如何编程。 此外,Python还具有免费开源、跨平台、面向对象、胶水语言等优点,在系统编程、图形界面开发、科学计算、Web开发、数据分析、人工智能等方面有广泛应用。 尤其是在数据分析和人工智能方面,Python已成为最受开发者欢迎的编程语言之一,不仅大量计算机专业人员选择使用Python进行快速开发,许多非计算机专业人员也纷纷选择Python语言来解决专业问题。 由于Python应用广泛,关于Python的参考书目前已经有很多,但将Python编程与数据分析、人工智
|
9天前
|
数据采集 存储 数据挖掘
Python网络爬虫实战:抓取并分析网页数据
使用Python的`requests`和`BeautifulSoup`,本文演示了一个简单的网络爬虫,抓取天气网站数据并进行分析。步骤包括发送HTTP请求获取HTML,解析HTML提取温度和湿度信息,以及计算平均温度。注意事项涉及遵守robots.txt、控制请求频率及处理动态内容。此基础爬虫展示了数据自动收集和初步分析的基础流程。【6月更文挑战第14天】
85 9
|
6天前
|
消息中间件 安全 开发者
Python global关键字分析
Python 是一种高级编程语言,拥有非常强大的功能和灵活性。在 Python 中,global 关键字可以用于在函数内部修改全局变量的值。本篇技术博客将介绍 global 关键字的使用方法和实现原理。
19 5
|
6天前
|
存储 Python
Python的命名空间和作用域分析
在Python中,命名空间(Namespace)是用来存储变量名和对象引用之间映射关系的字典,而作用域(Scope)是指程序中变量可以被访问的区域范围。Python中的命名空间是用来存储变量名和对象引用之间映射关系的字典,Python中存在3种命名空间:内置命名空间、全局命名空间和局部命名空间。Python中存在3种命名空间:内置命名空间、全局命名空间和局部命名空间。局部作用域:由局部命名空间定义,在函数内部定义的变量只能在该函数内部访问。在全局命名空间中定义的变量可以在模块内的任何函数或类中直接使用。
16 3
|
5天前
|
机器学习/深度学习 自然语言处理 数据可视化
文本挖掘与可视化:生成个性化词云的Python实践【7个案例】
词云(Word Cloud),又称为文字云或标签云,是一种用于文本数据可视化的技术,通过不同大小、颜色和字体展示文本中单词的出现频率或重要性。在词云中,更频繁出现的单词会显示得更大,反之则更小。
|
5天前
|
机器学习/深度学习 自然语言处理 数据可视化
文本挖掘与可视化:生成个性化词云的Python实践【7个案例】
词云是文本数据可视化的工具,显示单词频率,直观、美观,适用于快速展示文本关键信息。 - 用途包括关键词展示、数据探索、报告演示、情感分析和教育。 - 使用`wordcloud`和`matplotlib`库生成词云,`wordcloud`负责生成,`matplotlib`负责显示。 - 示例代码展示了从简单词云到基于蒙版、颜色和关键词权重的复杂词云生成。 - 案例覆盖了中文分词(使用`jieba`库)、自定义颜色和关键词权重的词云。 - 代码示例包括读取文本、分词、设置词云参数、显示和保存图像。
23 1
|
8天前
|
算法 搜索推荐 开发者
解锁Python代码的速度之谜:性能瓶颈分析与优化实践
探索Python性能优化,关注解释器开销、GIL、数据结构选择及I/O操作。使用cProfile和line_profiler定位瓶颈,通过Cython减少解释器影响,多进程避开GIL,优化算法与数据结构,以及借助asyncio提升I/O效率。通过精准优化,Python可应对高性能计算挑战。【6月更文挑战第15天】
15 1
|
10天前
|
Shell Python
GitHub星标破千Star!Python游戏编程的初学者指南
Python 是一种高级程序设计语言,因其简洁、易读及可扩展性日渐成为程序设计领域备受推崇的语言。 目前的编程书籍大多分为两种类型。第一种,与其说是教编程的书,倒不如说是在教“游戏制作软件”,或教授使用一种呆板的语言,使得编程“简单”到不再是编程。而第二种,它们就像是教数学课一样教编程:所有的原理和概念都以小的应用程序的方式呈现给读者。
|
1天前
|
Python
python datetime处理时间的详细分析
python datetime处理时间的详细分析
|
3天前
|
机器学习/深度学习 人工智能 前端开发
Python中的模块化编程
【6月更文挑战第17天】Python模块化编程与软件架构设计的关键在于拆分任务到独立模块,提高代码的可维护性、可重用性和可扩展性。例如,学生管理系统可分解为录入、查询和删除模块。MVC和MVVM架构模式有助于组织代码,而微服务和函数式编程将在未来发展中扮演重要角色。通过示例代码,读者能学习如何实现这些概念,提升项目开发效率和质量。
148 57