Python和tkinter实现的字母记忆配对游戏

Python和tkinter实现的字母记忆配对游戏

因为这个小游戏用到了tkinter,先简要介绍一下它。tkinter是Python的标准GUI(图形用户界面)库,它提供了一种简单而强大的方式来创建图形界面应用程序。它提供了创建基本图形界面所需的所有工具,同时保持了相对简单的学习曲线。tkinter是Python的内置库,无需额外安装。

messagebox是tkinter中用于创建各种类型的消息对话框的模块,需要注意的是messagebox是tkinter的一个子模块。为了正确使用messagebox,你需要从tkinter中单独导入它。

这个小游戏具有重新开始和难度设置功能。

“游戏”菜单,包含“新游戏”选项,点击它或完成一局游戏后,会自动开始新游戏。

“难度”菜单,难度设置,包含简单、中等和困难三个选项。

简单难度: 4x2 网格,8个方块

中等难度: 4x3 网格,12个方块

困难难度: 4x4 网格,16个方块

运行界面:

现在Python和tkinter实现字母记忆配对游戏源码,先看使用面向过程风格的版本源码:

import tkinter as tk
import tkinter.messagebox 
import random

def setup_menu(root, difficulty, new_game_func):
    """设置游戏菜单"""
    menubar = tk.Menu(root)
    root.config(menu=menubar)
    
    # 创建"游戏"菜单
    game_menu = tk.Menu(menubar, tearoff=0)
    menubar.add_cascade(label="游戏", menu=game_menu)
    game_menu.add_command(label="新游戏", command=new_game_func)
    
    # 创建"难度"菜单
    difficulty_menu = tk.Menu(menubar, tearoff=0)
    menubar.add_cascade(label="难度", menu=difficulty_menu)
    difficulty_menu.add_radiobutton(label="简单", variable=difficulty, value="简单", command=new_game_func)
    difficulty_menu.add_radiobutton(label="中等", variable=difficulty, value="中等", command=new_game_func)
    difficulty_menu.add_radiobutton(label="困难", variable=difficulty, value="困难", command=new_game_func)

def create_button(frame, row, col, on_click_func):
    """创建游戏按钮"""
    button = tk.Button(frame, text='', width=10, height=5, 
                       command=lambda: on_click_func(row, col))
    button.grid(row=row, column=col, padx=5, pady=5)
    return button

def new_game():
    """开始新游戏"""
    global matches_found, first_click, buttons, symbols
    
    matches_found = 0
    first_click = None
    
    # 清除旧的游戏布局
    for widget in frame.winfo_children():
        widget.destroy()
    
    buttons = []
    
    # 根据难度设置游戏布局和符号
    if difficulty.get() == "简单":
        rows, cols = 4, 2
        symbols = ['A', 'B', 'C', 'D'] * 2
    elif difficulty.get() == "中等":
        rows, cols = 4, 3
        symbols = ['A', 'B', 'C', 'D', 'E', 'F'] * 2
    else:  # 困难
        rows, cols = 4, 4
        symbols = ['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H'] * 2
    
    random.shuffle(symbols)
    
    # 创建游戏按钮
    for i in range(rows):
        for j in range(cols):
            button = create_button(frame, i, j, on_click)
            buttons.append(button)

def on_click(row, col):
    """处理按钮点击事件"""
    global first_click, matches_found
    
    index = row * len(frame.grid_slaves()) // 4 + col
    button = buttons[index]
    
    if button['text'] == '':
        button['text'] = symbols[index]
        
        if first_click is None:
            first_click = (index, button)
        else:
            if symbols[index] == first_click[1]['text']:
                matches_found += 1
                if matches_found == len(symbols) // 2:
                    tk.messagebox.showinfo("恭喜", "你赢了!")
                    new_game()
            else:
                # 如果不匹配,0.5秒后隐藏按钮
                root.after(500, hide_buttons, index, first_click[0])
            first_click = None

def hide_buttons(index1, index2):
    """隐藏不匹配的按钮"""
    buttons[index1]['text'] = ''
    buttons[index2]['text'] = ''

# 主程序
root = tk.Tk()
root.title("字母记忆配对游戏")
root.geometry("400x450")  # 设置窗口的宽度为400像素,高度为450像素

difficulty = tk.StringVar()
difficulty.set("简单")  # 默认难度为简单

frame = tk.Frame(root)
frame.pack()

setup_menu(root, difficulty, new_game)

# 初始化游戏变量
matches_found = 0
first_click = None
buttons = []
symbols = []

new_game()  # 开始第一局游戏

root.mainloop()  # 启动主事件循环

上面是使用面向过程风格的版本,下面改写为使用面向对象风格的版本,源码如下:

import tkinter as tk
import tkinter.messagebox 
import random

class MemoryGame:
    def __init__(self, master):
        self.master = master
        self.master.title("字母记忆配对游戏")
        self.master.geometry("400x450") #设置了窗体的宽度为400像素,高度为450像素
        
        self.buttons = []
        self.first_click = None
        self.matches_found = 0
        
        self.difficulty = tk.StringVar()
        self.difficulty.set("简单")
        
        self.setup_menu()
        self.setup_game()
        
    def setup_menu(self):
        menubar = tk.Menu(self.master)
        self.master.config(menu=menubar)
        
        game_menu = tk.Menu(menubar, tearoff=0)
        menubar.add_cascade(label="游戏", menu=game_menu)
        game_menu.add_command(label="新游戏", command=self.new_game)
        
        difficulty_menu = tk.Menu(menubar, tearoff=0)
        menubar.add_cascade(label="难度", menu=difficulty_menu)
        difficulty_menu.add_radiobutton(label="简单", variable=self.difficulty, value="简单", command=self.new_game)
        difficulty_menu.add_radiobutton(label="中等", variable=self.difficulty, value="中等", command=self.new_game)
        difficulty_menu.add_radiobutton(label="困难", variable=self.difficulty, value="困难", command=self.new_game)
        
    def setup_game(self):
        self.frame = tk.Frame(self.master)
        self.frame.pack()
        
        self.new_game()
        
    def new_game(self):
        self.matches_found = 0
        self.first_click = None
        
        for widget in self.frame.winfo_children():
            widget.destroy()
        
        self.buttons = []
        
        if self.difficulty.get() == "简单":
            rows, cols = 4, 2
            symbols = ['A', 'B', 'C', 'D'] * 2
        elif self.difficulty.get() == "中等":
            rows, cols = 4, 3
            symbols = ['A', 'B', 'C', 'D', 'E', 'F'] * 2
        else:  # 困难
            rows, cols = 4, 4
            symbols = ['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H'] * 2
        
        random.shuffle(symbols)
        
        for i in range(rows):
            for j in range(cols):
                button = tk.Button(self.frame, text='', width=10, height=5, 
                                   command=lambda x=i, y=j: self.on_click(x, y))
                button.grid(row=i, column=j, padx=5, pady=5)
                self.buttons.append(button)
        
        self.symbols = symbols
    
    def on_click(self, row, col):
        index = row * len(self.frame.grid_slaves()) // 4 + col
        button = self.buttons[index]
        
        if button['text'] == '':
            button['text'] = self.symbols[index]
            
            if self.first_click is None:
                self.first_click = (index, button)
            else:
                if self.symbols[index] == self.first_click[1]['text']:
                    self.matches_found += 1
                    if self.matches_found == len(self.symbols) // 2:
                        tk.messagebox.showinfo("恭喜", "你赢了!")
                        self.new_game()
                else:
                    # 如果不匹配,0.5秒后隐藏按钮
                    self.master.after(500, self.hide_buttons, index, self.first_click[0])
                self.first_click = None
    
    def hide_buttons(self, index1, index2):
        self.buttons[index1]['text'] = ''
        self.buttons[index2]['text'] = ''

root = tk.Tk()
game = MemoryGame(root)
root.mainloop()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/751722.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

生产者发送数据,kafka服务器接收数据异常的问题记录

现象: 某个客户要求审计日志用kafka的方式传输给他们,使用了第三方的librdkafka库来开发。 往客户提供的kafka服务器上的一个topic发送数据,这个topic有三个分区,客户反馈接收到的数据和发送端发送的实际数量对不上,他…

Elasticsearch环境搭建|ES单机|ES单节点模式启动|ES集群搭建|ES集群环境搭建

文章目录 版本选择单机ES安装与配置创建非root用户导入安装包安装包解压配置JDK环境变量配置single-node配置JVM参数后台启动|启动日志查看启动成功,访问终端访问浏览器访问 Kibana安装修改配置后台启动|启动日志查看浏览器访问 ES三节点集群搭建停止es服务域名配置…

平板WPS转换的PDF文件保存位置解析

在日常工作和生活中,我们经常需要将文档转换成PDF格式进行分享,以确保接收者能够无障碍地查看文件内容,不受软件版本或操作系统的限制。WPS作为一款功能强大的办公软件,也提供了文档转换为PDF的功能。然而,有时在转换并…

HarmonyOS--数据持久化--关系型数据库

文档中心 关系型数据库 场景介绍 关系型数据库基于SQLite组件,适用于存储包含复杂关系数据的场景,比如一个班级的学生信息,需要包括姓名、学号、各科成绩等,又或者公司的雇员信息,需要包括姓名、工号、职位等&#…

hnust 1817 算法10-10,10-11:堆排序

hnust 1817 算法10-10,10-11:堆排序 题目描述 堆排序是一种利用堆结构进行排序的方法,它只需要一个记录大小的辅助空间,每个待排序的记录仅需要占用一个存储空间。 首先建立小根堆或大根堆,然后通过利用堆的性质即堆顶的元素是最…

Mac14.1.2 M1芯片免费读写ntfs硬盘-亲测有效,免费!!!

1. 安装homebrew 打开终端,使用以下命令 /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)" 根据提示逐步完成即可,镜像选择我这里都是保持1的选项。 2. 重启终端 安装完成homebrew后,需…

Vite: 关于Rollup打包

概述 Rollup 是一款基于 ES Module 模块规范实现的 JavaScript 打包工具,在前端社区中赫赫有名,同时也在 Vite 的架构体系中发挥着重要作用不仅是 Vite 生产环境下的打包工具,其插件机制也被 Vite 所兼容,可以说是 Vite 的构建基…

单点登录(cookie+Redis)

1、什么是单点登录? Single Sign On简称SSo,只需要登录一次就可以在整个系统实现访问。 因为session的特性,是没有办法在多个服务系统之间实现数据的共享。 解决一个分布式session的问题。目前我们使用redis来实现分布式session。 1.1、新问题…

【数据结构】(C语言):队列

队列: 线性的集合。先进先出(FIFO,first in first out)。两个指针:头指针(指向第一个进入且第一个出去的元素),尾指针(指向最后一个进入且最后一个出去的元素&#xff0…

Redis优化之持久化

目录 1.Redis高可用 2.Redis持久化 2.1 RDB持久化 2.1.1 触发条件 2.1.2 执行流程 2.1.3 启动时加载 2.2 AOF持久化 2.2.1 开启AOF 2.2.2 执行流程 2.2.3 文件重写触发方式 2.2.4 文件重写的流程 2.2.5 启动时加载 2.3 RDB和AOF的优缺点 3.Redis性能管理 3.1 查看…

C++ 教程 - 07 类的静态成员

文章目录 静态成员 静态成员 使用static修饰的成员; 静态的成员变量; 仅保留一份副本,不管创建多少个实例对象,都共享这一份数据;类、对象均可以调用;类外重新声明,并通过类初始化;…

怎么在vite项目中全局导入一个scss文件

怎么在vite项目中全局导入一个scss文件 🎉🎉🎉欢迎来到我的博客,我是一名自学了2年半前端的大一学生,熟悉的技术是JavaScript与Vue.目前正在往全栈方向前进, 如果我的博客给您带来了帮助欢迎您关注我,我将会持续不断的更新文章!!!&#x1f64…

腾讯云CVM,CentOS8系统下部署Java-Web项目步骤详解

在CVM中部署项目首先要配置好JDK,Tomcat,Mysql(这里以Tomcat和Mysql为例)。部署JDK和Tomcat的步骤可以参考 CentOS7系统下部署tomcat,浏览器访问localhost:8080/_不积跬步,无以至千里;不积小流,无以成江河。-CSDN博客 我这里从Mysql的安装和设…

Java | Leetcode Java题解之第201题数字范围按位与

题目&#xff1a; 题解&#xff1a; class Solution {public int rangeBitwiseAnd(int m, int n) {while (m < n) {// 抹去最右边的 1n n & (n - 1);}return n;} }

C#——命名空间详情

命名空间 在 C# 中&#xff0c;可以将命名空间看作是一个范围&#xff0c;用来标注命名空间中成员的归属&#xff0c;一个命名空间中类与另一个命名空间中同名的类互不冲突&#xff0c;但在同一个命名空间中类的名称必须是唯一的。 定义命名空间 定义命名空间需要使用 namesp…

微软推出最新视觉基础模型Florence-2 可在浏览器运行

据微软官方消息&#xff0c;微软推出视觉基础模型Florence-2&#xff0c;该模型现已能够在支持WebGPU的浏览器中100%本地运行。Florence-2-base-ft是一个拥有2.3亿参数的视觉基础模型&#xff0c;采用基于提示的方法来处理广泛的视觉和视觉语言任务。 该模型支持多种功能&…

youlai-boot项目的学习(4) 前后端本地部署

环境 1、macOS, brew, IntelliJ IDEA, WebStrom 2、后端&#xff1a;https://gitee.com/youlaiorg/youlai-boot.git , master, 9a753a2e94985ed4cbbf214156ca035082e02723 3、前端&#xff1a;https://gitee.com/youlaiorg/vue3-element-admin.git, master, 66b913ef01dc880ad…

25届最近5年重庆邮电大学自动化考研院校分析

重庆邮电大学 目录 一、学校学院专业简介 二、考试科目指定教材 三、近5年考研分数情况 四、近5年招生录取情况 五、最新一年分数段图表 六、历年真题PDF 七、初试大纲复试大纲 八、学费&奖学金&就业方向 一、学校学院专业简介 二、考试科目指定教材 1、考试…

提取url中的参数

let url https://alibaba.com?a1&b2&c3#hash function queryUrlParams(URL){let url URL.split(?)[1];const urlSearchParams new URLSearchParams(url);console.log(url1, urlSearchParams);console.log(entries,urlSearchParams.entries())const params Object…

大模型推理知识总结

一、大模型推理概念 大多数流行的only-decode LLM&#xff08;例如 GPT-3&#xff09;都是针对因果建模目标进行预训练的&#xff0c;本质上是作为下一个词预测器。这些 LLM 将一系列tokens作为输入&#xff0c;并自回归生成后续tokens&#xff0c;直到满足停止条件&#xff0…