Python+tkinter实现2048游戏

游戏规则

2048 是一个单人拼图游戏,目标是通过滑动方块将相同数字的方块合并,最终得到一个数字为 2048 的方块。

  • 游戏在一个 4x4 的网格上进行,初始时有两个方块,每个方块的值为 2 或 4。
  • 玩家可以通过上下左右方向键滑动方块,所有方块会向滑动方向移动,直到被其他方块或网格边界阻挡。
  • 当两个相同数字的方块碰撞时,它们会合并成一个新的方块,值为原来两个方块的和。例如,两个值为 2 的方块合并成一个值为 4 的方块。每次合并方块时,合并后的方块数值会加到玩家的总得分中。
  • 每次滑动后,会在一个空格子中随机生成一个新的方块,值为 2 或 4。
  • 当玩家成功生成一个值为 2048 的方块时,游戏胜利。
  • 当网格中没有空格子且无法进行任何合并操作时,游戏结束。
  • 玩家可以查看排行榜,显示历史最高的十次成绩。

游戏代码(使用 tkinter 库创建图形用户界面)

import tkinter as tk
import random
import osclass Game2048:def __init__(self, master):self.master = masterself.master.title("2048 Game")self.center_window(400, 450)  # 设置窗口居中# 初始化游戏板,4x4的矩阵,初始值为0self.board = [[0] * 4 for _ in range(4)]self.score = 0  # 初始化分数self.create_widgets()self.reset_game()# 绑定键盘事件self.master.bind("<Key>", self.key_handler)def center_window(self, width, height):# 获取屏幕宽度和高度screen_width = self.master.winfo_screenwidth()screen_height = self.master.winfo_screenheight()# 计算窗口左上角的位置x = (screen_width - width) // 2y = (screen_height - height) // 2self.master.geometry(f'{width}x{height}+{x}+{y}')def create_widgets(self):# 创建一个顶部框架用于放置按钮和记分牌top_frame = tk.Frame(self.master)top_frame.pack(fill=tk.X, pady=10)# 创建排行榜按钮leaderboard_button = tk.Button(top_frame, text="Leaderboard", command=self.show_leaderboard)leaderboard_button.pack(side=tk.RIGHT, padx=10)# 创建重启按钮restart_button = tk.Button(top_frame, text="Restart", command=self.reset_game)restart_button.pack(side=tk.RIGHT, padx=10)# 创建记分牌self.score_label = tk.Label(top_frame, text=f"Score: {self.score}", font=("Helvetica", 12))self.score_label.pack(side=tk.RIGHT, padx=10)# 创建一个400x400的画布self.canvas = tk.Canvas(self.master, width=400, height=400, bg="white")self.canvas.pack(pady=0)  # 调整pady参数self.draw_grid()def draw_grid(self):# 定义不同数字对应的颜色colors = {0: "lightgray",2: "lightyellow",4: "lightgoldenrod",8: "orange",16: "darkorange",32: "tomato",64: "red",128: "yellow",256: "gold",512: "lightgreen",1024: "green",2048: "darkgreen"}# 绘制4x4的网格for i in range(4):for j in range(4):x0, y0 = i * 100, j * 100x1, y1 = x0 + 100, y0 + 100value = self.board[j][i]color = colors.get(value, "black")  # 获取对应颜色,默认黑色self.canvas.create_rectangle(x0, y0, x1, y1, fill=color)if value != 0:# 在非零的格子中间绘制数字self.canvas.create_text(x0 + 50, y0 + 50, text=str(value), font=("Helvetica", 24), fill="black")def reset_game(self):# 重置游戏板和分数self.board = [[0] * 4 for _ in range(4)]self.score = 0self.update_score()# 添加两个初始的随机方块self.add_new_tile()self.add_new_tile()self.update_ui()def add_new_tile(self):# 找到所有空的格子empty_cells = [(i, j) for i in range(4) for j in range(4) if self.board[i][j] == 0]if empty_cells:# 随机选择一个空格子,并放入2或4i, j = random.choice(empty_cells)self.board[i][j] = 2 if random.random() < 0.9 else 4def update_ui(self):# 更新UI,重新绘制网格self.canvas.delete("all")self.draw_grid()def update_score(self):# 更新记分牌self.score_label.config(text=f"Score: {self.score}")def key_handler(self, event):# 处理键盘事件if event.keysym in ["Up", "Down", "Left", "Right"]:if self.move(event.keysym):# 如果移动成功,添加一个新的随机方块self.add_new_tile()self.update_ui()# 检查游戏是否结束或胜利if self.check_win():self.win()elif self.check_game_over():self.game_over()def move(self, direction):# 滑动并合并行或列def slide(row):new_row = [i for i in row if i != 0]for i in range(len(new_row) - 1):if new_row[i] == new_row[i + 1]:new_row[i] *= 2self.score += new_row[i]  # 更新分数new_row[i + 1] = 0new_row = [i for i in new_row if i != 0]return new_row + [0] * (4 - len(new_row))moved = Falseif direction == "Up":for j in range(4):col = [self.board[i][j] for i in range(4)]new_col = slide(col)for i in range(4):if self.board[i][j] != new_col[i]:moved = Trueself.board[i][j] = new_col[i]elif direction == "Down":for j in range(4):col = [self.board[i][j] for i in range(4)]new_col = slide(col[::-1])[::-1]for i in range(4):if self.board[i][j] != new_col[i]:moved = Trueself.board[i][j] = new_col[i]elif direction == "Left":for i in range(4):new_row = slide(self.board[i])if self.board[i] != new_row:moved = Trueself.board[i] = new_rowelif direction == "Right":for i in range(4):new_row = slide(self.board[i][::-1])[::-1]if self.board[i] != new_row:moved = Trueself.board[i] = new_rowif moved:self.update_score()  # 更新分数return moveddef check_win(self):# 检查是否达到2048for row in self.board:if 2048 in row:return Truereturn Falsedef check_game_over(self):# 检查游戏是否结束for i in range(4):for j in range(4):if self.board[i][j] == 0:return Falseif i < 3 and self.board[i][j] == self.board[i + 1][j]:return Falseif j < 3 and self.board[i][j] == self.board[i][j + 1]:return Falsereturn Truedef game_over(self):# 显示游戏结束信息self.canvas.create_text(200, 200, text="Game Over", font=("Helvetica", 36), fill="red")self.save_score()  # 保存分数到文件def win(self):# 显示胜利信息self.canvas.create_text(200, 200, text="You win!", font=("Helvetica", 36), fill="green")self.save_score()  # 保存分数到文件def save_score(self):# 保存当前分数到文件with open("scores.txt", "a") as file:file.write(f"{self.score}\n")def show_leaderboard(self):# 读取分数并显示排行榜if os.path.exists("scores.txt"):with open("scores.txt", "r") as file:scores = [int(line.strip()) for line in file.readlines()]scores.sort(reverse=True)top_scores = scores[:10]else:top_scores = []# 创建一个新的窗口显示排行榜leaderboard_window = tk.Toplevel(self.master)leaderboard_window.title("Leaderboard")leaderboard_window.geometry("200x250")tk.Label(leaderboard_window, text="Top 10 Scores", font=("Helvetica", 16)).pack()for idx, score in enumerate(top_scores, 1):tk.Label(leaderboard_window, text=f"{idx}. {score}", font=("Helvetica", 14)).pack()if __name__ == "__main__":root = tk.Tk()game = Game2048(root)root.mainloop()

页面效果

在这里插入图片描述
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

微服务间调用

一、restTemplate 1、先将restTemplate注册成为一个bean Configuration public class RemoteCallConfig {Beanpublic RestTemplate restTemplate() {return new RestTemplate();} }2、实现代码 private void handleCartItems(List<CartVO> vos) {// TODO 1.获取商品id…

【C++ Primer Plus习题】6.4

问题: 解答: #include <iostream> using namespace std;const int strsize 40; const int usersize 40;typedef struct _Bop {char fullname[strsize];char title[strsize];char bopname[strsize];int preference; }Bop;Bop bop_user[usersize] {{"Wimp Macho&q…

使用Python调用JavaScript进行网页自动化操作

随着互联网技术的飞速发展&#xff0c;网页自动化操作在数据抓取、用户界面测试、内容管理等多个领域变得越来越重要。Python作为一种流行的编程语言&#xff0c;因其简洁的语法和强大的库支持&#xff0c;成为了许多开发者进行网页自动化的首选工具。然而&#xff0c;面对动态…

【C++ Primer Plus习题】6.8

问题: 解答: #include <iostream> #include <fstream> #include <string> using namespace std;int main() {string filename;ifstream stream;char read_char;int count0;cout << "请输入要打开的文件:";getline(cin, filename);stream.op…

微分方程(Blanchard Differential Equations 4th)中文版Section5.1

平衡点分析 从第3章的工作中,我们能够对线性系统的解有定性和解析的理解。不幸的是,非线性系统通常不容易使用我们开发的解析和代数技术来分析,但我们可以利用线性系统的数学来理解非线性系统在其平衡点附近的行为。 Van der Pol 方程 为了说明如何分析平衡点附近解的行为…

输电线路分布式故障诊断系统:分布式智慧网络的构建

输电线路分布式故障诊断系统&#xff1a;分布式智慧网络的构建 今天&#xff0c;就让深圳鼎信智慧科技陪大家一起走进输电线路分布式故障定位系统的世界&#xff1a; 1、系统架构&#xff1a;分布式智慧网络的构建 输电线路分布式故障定位系统主要由三大核心部分组成&#x…

数据结构(6.4_2)——最短路径问题_BFS算法

最短路径问题 BFS求无权图的单源最短路径 原代码 改造visit函数后

Elastic日志分析

目录 介绍步骤 介绍 Elasticsearch 是在 Apache Lucene 上构建的分布式搜索和分析引擎。Elasticsearch常用于日志分析、全文搜索、安全智能、业务分析和运维智能使用案例。 可以使用 JSON 文档形式或通过 API 等将数据发送到 Elasticsearch。 Elasticsearch 自动存储原始文档…

Launcher3 长按Hotseat图标,显示删除角标(红底白杠杠用于删除图标或者显示应用未读消息数量)

基于Android 13,Launcher3实现需求&#xff1a; 1. 长按Hotseat的图标显示红色删除角标 2. 点击角标&#xff0c;删除图标并保存到Database 3.点击其他地方&#xff0c;取消编辑hotseat图标模式 实现效果&#xff1a; 实现原理&#xff1a; 图标是由BubbleTextView来是实现…

数据库系统入门指南

数据库系统入门指南 数据库系统是现代信息技术的重要组成部分。本文将介绍数据库、数据库管理系统、数据库应用系统以及相关概念&#xff0c;帮助初学者快速上手。 什么是数据库&#xff1f; 数据库是一个有组织、可共享的数据集合。它存储在计算机中&#xff0c;数据按一定…

idea import配置

简介 本文记录idea中import相关配置&#xff1a;自动导入依赖、自动删除无用依赖、避免自动导入*包 自动导入依赖 在编辑代码时&#xff0c;当只有一个具有匹配名称的可导入声明时&#xff0c;会自动添加导入 File -> Settings -> Editor -> General -> Auto Imp…

FL Studio24苹果mac电脑破解绿色版安装包下载

FL Studio 24最新版本&#xff0c;这可不仅仅是一个音乐制作软件的升级&#xff0c;它是音乐创作爱好者的福音&#xff0c;是专业制作人的心头好。那么&#xff0c;它究竟有哪些魔力&#xff0c;能让这么多人为之疯狂呢&#xff1f; 我们来看看它的界面。FL Studio 24的界面设…

Nginx: 负载均衡基础配置, 加权轮序, hash算法, ip_hash算法, least_conn算法

负载均衡 在真正的反向代理场景中&#xff0c;必然涉及到的一个概念&#xff0c;就是负载均衡所谓负载均衡&#xff0c;也就是将Nginx的请求发送给后端的多台应用程序服务器通常的应用程序服务器&#xff0c;后面的每台服务器都是一个同等的角色&#xff0c;提供相同的功能 用…

类与ES6类之间的继承

前言 ● 下面是之前学习ES6 classes的代码 class PersonCl {constructor(fullName, birthYear) {this.fullName fullName;this.birthYear birthYear;}calcAge() {console.log(2037 - this.birthYear);}greet() {console.log(你好${this.fullName});}get age() {return 2037…

debian12 - openssh-9.6.P1的编译安装(真机 - 联想G480)

文章目录 debian12 - openssh-9.6.P1的编译安装(真机 - 联想G480)概述笔记G480上安装debian12配置debian12现在用WindTerm_2.6.0按照telnet方式去连接试试配置debian12中的telnet安装telnet服务查看所有服务当前ssh, telnet状态准备更新openssl3.2和openssh在真机上更新openssl…

自动分词代码

代码 from wordsegment import load, segment# 加载模型 load()# 示例 actions ["seeyoulater","turnleft","turnr" ]segmented_actions [segment(action) for action in actions] segmented_actions [" ".join(action) for actio…

云计算概述

云计算的产生以及发展 分布式计算&#xff1a;包含了云计算和网格计算 云计算&#xff1a;以数据为中心进行的计算 网格计算&#xff1a;以计算为中心进行的计算 诞生-1999 初期的发展-2007-2008 加速发展-2009-2014 日渐成熟阶段-2015-目前 云计算的种类 公有云-第三方提供…

chapter08-面向对象编程——(chapter08作业)——day10

343-作业01 package chapter08.homeworks;public class Homework01 {public static void main(String[] args) {/*1.定义一个Person类{name, age, job},初始化Person对象数组&#xff0c;有3个person对象&#xff0c;并按照age从大到小进行排序,提示&#xff0c;使用冒泡排序 H…

Day50 | 108.冗余连接 109.冗余连接II

108.冗余连接 108. 冗余连接 题目 题目描述 树可以看成是一个图&#xff08;拥有 n 个节点和 n - 1 条边的连通无环无向图&#xff09;。 现给定一个拥有 n 个节点&#xff08;节点标号是从 1 到 n&#xff09;和 n 条边的连通无向图&#xff0c;请找出一条可以删除的边&…

一个简单的CRM客户信息管理系统,提供客户,线索,公海,联系人,跟进信息和数据统计功能(附源码)

前言 在当今快速发展的商业环境中&#xff0c;企业管理面临着种种挑战&#xff0c;尤其是如何有效管理和维护客户关系成为了一个关键问题。现有的一些处理方案往往存在功能分散、操作复杂、数据孤岛等痛点&#xff0c;这不仅影响了工作效率&#xff0c;也难以满足企业对客户关…