当前位置: 首页 > news >正文

使用Pygame制作“俄罗斯方块”游戏

1. 前言

俄罗斯方块(Tetris) 是一款由方块下落、行消除等核心规则构成的经典益智游戏:

  • 每次从屏幕顶部出现一个随机的方块(由若干小方格组成),玩家可以左右移动或旋转该方块,让它合适地堆叠在底部或其他方块之上。
  • 当某一行被填满时,该行会被消除,并给玩家增加分数。
  • 若方块叠加到顶部无法容纳新的方块,就表示游戏结束。

在本篇中,我们将使用 PythonPygame 来实现一个简化版的俄罗斯方块,主要演示核心流程,帮助你掌握更多 2D 游戏编程的常见技巧。


2. 开发环境

  1. Python 3.x
  2. Pygame:如果尚未安装,请执行
    pip install pygame
    
  3. 支持图形界面的桌面操作系统(Windows、macOS 或大部分 Linux)。

3. 游戏思路与要点

俄罗斯方块主要有以下几个关键逻辑点:

  1. 网格与方块

    • 整个游戏区域可被拆分为若干行列(比如 10 列 × 20 行)。
    • 下落的“方块”通常由四个小方格(称为 Tetromino)组成,常见形状有 I、O、T、S、Z、J、L 七种。
  2. 方块的移动和旋转

    • 在每帧或固定时间间隔,让方块自动向下移动一次。
    • 监听玩家按键:左右移动、加速下落、旋转方块等。
    • 在方块运动前要检测是否与已有方块发生碰撞或超出边界;若无法移动,则保持在原处。
  3. 行检测与消除

    • 每次放置完一个方块后,需要检查游戏网格中是否有某一行被填满(全部不是空)。
    • 若有满行,则消除该行,并将上方的所有行整体下移相应的行数,为玩家增加分数。
  4. 游戏结束检测

    • 当新的方块在顶部生成时,若该区域已被占满而无法容纳,即表示游戏结束。
  5. 随机性与关卡速度

    • 生成方块形状时,通常随机从七种基本形状中选择。
    • 随着分数的提升,可以让方块下落的速度越来越快,增强挑战性。

4. 完整示例代码

以下代码示例仅实现一个简易版本的俄罗斯方块,展示最核心的逻辑。你可以将其保存为 tetris_game.py 并运行,进一步研究与改进。

import pygame
import sys
import random# 初始化 pygame
pygame.init()# --------------------------
# 配置参数
# --------------------------
CELL_SIZE = 30       # 每个网格的像素大小
COLS = 10            # 游戏区域的列数
ROWS = 20            # 游戏区域的行数WINDOW_WIDTH = CELL_SIZE * COLS
WINDOW_HEIGHT = CELL_SIZE * ROWS
FPS = 30             # 帧率# 颜色
BLACK = (0, 0, 0)
WHITE = (255, 255, 255)
GRAY  = (128, 128, 128)
RED   = (255, 0, 0)
GREEN = (0, 255, 0)
BLUE  = (0, 0, 255)
CYAN  = (0, 255, 255)
YELLOW = (255, 255, 0)
MAGENTA = (255, 0, 255)
ORANGE  = (255, 165, 0)# 为了区分不同形状,定义一个全局形状/颜色映射
SHAPES_COLORS = [([[1, 1, 1, 1]], CYAN),       # I 形 (1行4列)([[1, 1],[1, 1]], YELLOW),          # O 形 (2行2列)([[0, 1, 0],[1, 1, 1]], MAGENTA),      # T 形([[1, 1, 0],[0, 1, 1]], GREEN),        # S 形([[0, 1, 1],[1, 1, 0]], RED),          # Z 形([[1, 0, 0],[1, 1, 1]], BLUE),         # J 形([[0, 0, 1],[1, 1, 1]], ORANGE)        # L 形
]# 创建窗口
screen = pygame.display.set_mode((WINDOW_WIDTH, WINDOW_HEIGHT))
pygame.display.set_caption("俄罗斯方块 - Pygame")
clock = pygame.time.Clock()# 字体
font = pygame.font.SysFont("arial", 24)# --------------------------
# 游戏网格
# --------------------------
# 用一个 2D 列表保存网格信息:0 表示空,其他表示对应的颜色
grid = [[0] * COLS for _ in range(ROWS)]def draw_text(surface, text, color, x, y):"""在指定位置绘制文字"""label = font.render(text, True, color)surface.blit(label, (x, y))def draw_grid(surface):"""绘制当前网格状态"""for r in range(ROWS):for c in range(COLS):val = grid[r][c]rect = pygame.Rect(c * CELL_SIZE, r * CELL_SIZE, CELL_SIZE, CELL_SIZE)if val == 0:pygame.draw.rect(surface, BLACK, rect, 0)  # 空 -> 绘制黑色背景pygame.draw.rect(surface, GRAY, rect, 1)   # 网格线else:# 如果 val 非 0,则 val 存储的是颜色pygame.draw.rect(surface, val, rect, 0)pygame.draw.rect(surface, GRAY, rect, 1)def check_lines():"""检测并消除已填满的行,返回消除行数"""full_lines = 0for r in range(ROWS):if 0 not in grid[r]:  # 如果这一行没有空格full_lines += 1# 将该行之上的所有行往下移动for rr in range(r, 0, -1):grid[rr] = grid[rr - 1][:]# 最上面那行设为空grid[0] = [0] * COLSreturn full_linesclass Tetromino:"""表示当前下落的方块包含:shape  -> 形状(一个2D list)color  -> 颜色row, col -> 在grid中的坐标(左上角)"""def __init__(self, shape, color):self.shape = shapeself.color = colorself.row = 0self.col = COLS // 2 - len(shape[0]) // 2def width(self):return len(self.shape[0])def height(self):return len(self.shape)def can_move(self, dr, dc):"""判断方块是否能移动dr, dc"""new_row = self.row + drnew_col = self.col + dcfor r in range(self.height()):for c in range(self.width()):if self.shape[r][c] != 0:rr = new_row + rcc = new_col + c# 判断是否越界if rr < 0 or rr >= ROWS or cc < 0 or cc >= COLS:return False# 判断是否和已有方块重叠if grid[rr][cc] != 0:return Falsereturn Truedef move(self, dr, dc):"""执行移动"""if self.can_move(dr, dc):self.row += drself.col += dcreturn Truereturn Falsedef rotate(self):"""顺时针旋转90度先生成新形状,然后判断是否能放置"""rotated = list(zip(*self.shape[::-1]))  # 转置 + 逆序 可实现顺时针旋转new_shape = [list(row) for row in rotated]# 临时保存原 shapeold_shape = self.shapeself.shape = new_shape# 如果旋转后出界或冲突,则恢复原 shapeif not self.can_move(0, 0):self.shape = old_shapedef lock(self):"""当方块无法继续移动后,锁定到网格"""for r in range(self.height()):for c in range(self.width()):if self.shape[r][c] != 0:grid[self.row + r][self.col + c] = self.colordef new_tetromino():"""随机生成一个新的方块对象"""shape, color = random.choice(SHAPES_COLORS)return Tetromino(shape, color)def is_game_over():"""判断顶端是否已堆满"""for c in range(COLS):if grid[0][c] != 0:return Truereturn Falsedef main():current_piece = new_tetromino()next_piece = new_tetromino()fall_time = 0fall_speed = 0.5  # 每0.5秒下落一格score = 0running = Truewhile running:dt = clock.tick(FPS) / 1000  # 以秒为单位的帧间隔fall_time += dt# 1) 事件处理for event in pygame.event.get():if event.type == pygame.QUIT:running = Falseelif event.type == pygame.KEYDOWN:if event.key == pygame.K_LEFT:current_piece.move(0, -1)elif event.key == pygame.K_RIGHT:current_piece.move(0, 1)elif event.key == pygame.K_DOWN:# 快速下落current_piece.move(1, 0)elif event.key == pygame.K_UP:# 旋转current_piece.rotate()# 2) 自动下落if fall_time >= fall_speed:if not current_piece.move(1, 0):# 不能再下落 -> 锁定到 gridcurrent_piece.lock()# 检查是否有行被消除lines_cleared = check_lines()score += lines_cleared * 10# 判断游戏是否结束if is_game_over():running = Falseelse:# 生成新的方块current_piece = next_piecenext_piece = new_tetromino()fall_time = 0# 3) 绘制screen.fill(BLACK)draw_grid(screen)# 预先“绘制”当前方块(仅在可见区域内)for r in range(current_piece.height()):for c in range(current_piece.width()):if current_piece.shape[r][c] != 0:rr = current_piece.row + rcc = current_piece.col + cif rr >= 0:rect = pygame.Rect(cc * CELL_SIZE, rr * CELL_SIZE, CELL_SIZE, CELL_SIZE)pygame.draw.rect(screen, current_piece.color, rect, 0)pygame.draw.rect(screen, GRAY, rect, 1)draw_text(screen, f"Score: {score}", WHITE, 10, 10)pygame.display.flip()game_over(screen, score)def game_over(surface, score):"""游戏结束画面"""surface.fill(GRAY)draw_text(surface, "Game Over!", WHITE, WINDOW_WIDTH // 2 - 60, WINDOW_HEIGHT // 2 - 30)draw_text(surface, f"Your Score: {score}", WHITE, WINDOW_WIDTH // 2 - 70, WINDOW_HEIGHT // 2 + 10)pygame.display.flip()pygame.time.wait(3000)pygame.quit()sys.exit()if __name__ == "__main__":main()

核心逻辑解析

  1. grid 用于存储当前游戏区域的状态

    • grid[r][c] == 0 表示空格。
    • 其他值表示已经堆积在那儿的方块颜色(例如 (255,0,0) 对应红色等)。
  2. 方块(Tetromino) 类

    • 通过 shape(2D List)和 color 来描述该方块是什么形状、什么颜色。
    • row, col 表示方块左上角在网格坐标系的位置。
    • move(dr, dc) 会先检测移动后是否出界或与其他方块冲突;若可行再更新位置。
    • rotate() 通过矩阵转置 + 逆序实现顺时针旋转;若旋转后冲突,则恢复原状。
    • lock() 在方块无法下落后,将形状“锁定”到 grid 上。
  3. 行消除

    • check_lines() 判断每一行是否已被填满;若是,则把上方的行依次向下移动,并清空最上面的一行。
  4. 自动下落与输入控制

    • 用一个计时器 fall_time 不断累加帧间隔,达到 fall_speed 时让方块往下移动一次。
    • 监听左右键和上下键,实现左右移动、加速下落与旋转。
  5. 生成和切换方块

    • 每次锁定当前方块后,都从预先生成的 next_piece 切换过来,并随机生成下一个 next_piece
    • 这样可以在界面显示“下一块”预览(若想添加此功能,只需在界面上额外绘制 next_piece 的形状)。

5. 运行效果

image.png


6. 总结

通过这篇文章,我们用 Python + Pygame 实现了一个简化版的俄罗斯方块。这个项目涵盖了网格管理、方块碰撞与旋转、行检测与消除等常见的游戏逻辑。在此基础上,你完全可以自行改造并添加各种细节与高级功能,让游戏更贴近于真正的 Tetris。

相关文章:

使用Pygame制作“俄罗斯方块”游戏

1. 前言 俄罗斯方块&#xff08;Tetris&#xff09; 是一款由方块下落、行消除等核心规则构成的经典益智游戏&#xff1a; 每次从屏幕顶部出现一个随机的方块&#xff08;由若干小方格组成&#xff09;&#xff0c;玩家可以左右移动或旋转该方块&#xff0c;让它合适地堆叠在…...

【Block总结】ODConv动态卷积,适用于CV任务|即插即用

一、论文信息 论文标题&#xff1a;Omni-Dimensional Dynamic Convolution作者&#xff1a;Chao Li, Aojun Zhou, Anbang Yao发表会议&#xff1a;ICLR 2022论文链接&#xff1a;https://arxiv.org/pdf/2209.07947GitHub链接&#xff1a;https://github.com/OSVAI/ODConv 二…...

RK3568 opencv播放视频

文章目录 一、opencv相关视频播放类1. `cv::VideoCapture` 类主要构造方法:主要方法:2. 视频播放基本流程代码示例:3. 获取和设置视频属性4. 结合 FFmpeg 使用5. OpenCV 视频播放的局限性6. 结合 Qt 实现更高级的视频播放总结二、QT中的代码实现一、opencv相关视频播放类 在…...

《LLM大语言模型+RAG实战+Langchain+ChatGLM-4+Transformer》

文章目录 Langchain的定义Langchain的组成三个核心组件实现整个核心组成部分 为什么要使用LangchainLangchain的底层原理Langchain实战操作LangSmithLangChain调用LLM安装openAI库-国内镜像源代码运行结果小结 使用Langchain的提示模板部署Langchain程序安装langserve代码请求格…...

【搜索回溯算法篇】:拓宽算法视野--BFS如何解决拓扑排序问题

✨感谢您阅读本篇文章&#xff0c;文章内容是个人学习笔记的整理&#xff0c;如果哪里有误的话还请您指正噢✨ ✨ 个人主页&#xff1a;余辉zmh–CSDN博客 ✨ 文章所属专栏&#xff1a;搜索回溯算法篇–CSDN博客 文章目录 一.广度优先搜索&#xff08;BFS&#xff09;解决拓扑排…...

计算机网络 (61)移动IP

前言 移动IP&#xff08;Mobile IP&#xff09;是由Internet工程任务小组&#xff08;Internet Engineering Task Force&#xff0c;IETF&#xff09;提出的一个协议&#xff0c;旨在解决移动设备在不同网络间切换时的通信问题&#xff0c;确保移动设备可以在离开原有网络或子网…...

Elasticsearch+kibana安装(简单易上手)

下载ES( Download Elasticsearch | Elastic ) 将ES安装包解压缩 解压后目录如下: 修改ES服务端口&#xff08;可以不修改&#xff09; 启动ES 记住这些内容 验证ES是否启动成功 下载kibana( Download Kibana Free | Get Started Now | Elastic ) 解压后的kibana目…...

音视频多媒体编解码器基础-codec

如果要从事编解码多媒体的工作&#xff0c;需要准备哪些更为基础的内容&#xff0c;这里帮你总结完。 因为数据类型不同所以编解码算法不同&#xff0c;分为图像、视频和音频三大类&#xff1b;因为流程不同&#xff0c;可以分为编码和解码两部分&#xff1b;因为编码器实现不…...

【算法与数据结构】动态规划

目录 基本概念 最长递增子序列&#xff08;中等&#xff09; 最大子数组和&#xff08;中等&#xff09; 基本概念 重叠子问题 一个问题可以被分解为多个子问题&#xff0c;并且这些子问题在求解过程中会被多次重复计算。例如&#xff0c;在计算斐波那契数列时&#xff0c;…...

DeepSeekMoE:迈向混合专家语言模型的终极专业化

一、结论写在前面 论文提出了MoE语言模型的DeepSeekMoE架构&#xff0c;目的是实现终极的专家专业化(expert specialization)。通过细粒度的专家分割和共享专家隔离&#xff0c;DeepSeekMoE相比主流的MoE架构实现了显著更高的专家专业化和性能。从较小的2B参数规模开始&#x…...

什么是Maxscript?为什么要学习Maxscript?

MAXScript是Autodesk 3ds Max的内置脚本语言,它是一种与3dsMax对话并使3dsMax执行某些操作的编程语言。它是一种脚本语言,这意味着您不需要编译代码即可运行。通过使用一系列基于文本的命令而不是使用UI操作,您可以完成许多使用UI操作无法完成的任务。 Maxscript是一种专有…...

HyperLogLog 近似累计去重技术解析:大数据场景下的高效基数统计

目录 引言 一、HyperLogLog 核心原理 1.1 算法思想 1.2 误差特性 二、SQL 实现详解(PostgreSQL 示例)...

LabVIEW透镜多参数自动检测系统

在现代制造业中&#xff0c;提升产品质量检测的自动化水平是提高生产效率和准确性的关键。本文介绍了一个基于LabVIEW的透镜多参数自动检测系统&#xff0c;该系统能够在单一工位上完成透镜的多项质量参数检测&#xff0c;并实现透镜的自动搬运与分选&#xff0c;极大地提升了检…...

MySQL数据库(二)- SQL

目录 ​编辑 一 DDL (一 数据库操作 1 查询-数据库&#xff08;所有/当前&#xff09; 2 创建-数据库 3 删除-数据库 4 使用-数据库 (二 表操作 1 创建-表结构 2 查询-所有表结构名称 3 查询-表结构内容 4 查询-建表语句 5 添加-字段名数据类型 6 修改-字段数据类…...

【Block总结】HiLo注意力,局部自注意力捕获细粒度的高频信息,通过全局注意力捕获低频信息|即插即用

一、论文信息 标题: Fast Vision Transformers with HiLo AttentionGitHub链接: https://github.com/ziplab/LITv2论文链接: arXiv 二、创新点 HiLo注意力机制: 本文提出了一种新的自注意力机制——HiLo注意力&#xff0c;旨在同时捕捉图像中的高频和低频特征。该机制通过将…...

python 使用Whisper模型进行语音翻译

目录 一、Whisper 是什么? 二、Whisper 的基本命令行用法 三、代码实践 四、是否保留Token标记 五、翻译长度问题 六、性能分析 一、Whisper 是什么? Whisper 是由 OpenAI 开源的一个自动语音识别(Automatic Speech Recognition, ASR)系统。它的主要特点是: 多语言…...

C# Winform enter键怎么去关联button

1.关联按钮上的Key事件按钮上的keypress&#xff0c;keydown&#xff0c;keyup事件随便一个即可private void textBox1_KeyDown(object sender, KeyEventArgs e){if (e.KeyCode Keys.Enter){this.textBox2.Focus();}}2.窗体上的事件private void textBox2_KeyPress(object sen…...

Github 2025-01-30 Go开源项目日报 Top10

根据Github Trendings的统计,今日(2025-01-30统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Go项目10Ollama: 本地大型语言模型设置与运行 创建周期:248 天开发语言:Go协议类型:MIT LicenseStar数量:42421 个Fork数量:2724 次关注人…...

电路研究9.2.6——合宙Air780EP中HTTP——HTTP GET 相关命令使用方法研究

这个也是一种协议类型&#xff1a; 14.16 使用方法举例 根据之前多种类似的协议的相关信息&#xff1a; HTTP/HTTPS&#xff1a;超文本传输协议&#xff08;HTTP&#xff09;用于Web数据的传输&#xff0c;而HTTPS是HTTP的安全版本&#xff0c;使用SSL/TLS进行加密。与FTP相比&…...

Java手写简单Merkle树

Java手写Merkle树代码 package com.blockchain.qgy.component;import com.blockchain.qgy.model.MerkleTreeNode; import com.blockchain.qgy.util.SHAUtil;import java.util.*;public class MerkleTree<T> {//merkle树private List<MerkleTreeNode<T>> lis…...

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇&#xff0c;在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下&#xff1a; 【Note】&#xff1a;如果你已经完成安装等操作&#xff0c;可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作&#xff0c;重…...

pam_env.so模块配置解析

在PAM&#xff08;Pluggable Authentication Modules&#xff09;配置中&#xff0c; /etc/pam.d/su 文件相关配置含义如下&#xff1a; 配置解析 auth required pam_env.so1. 字段分解 字段值说明模块类型auth认证类模块&#xff0c;负责验证用户身份&am…...

spring:实例工厂方法获取bean

spring处理使用静态工厂方法获取bean实例&#xff0c;也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下&#xff1a; 定义实例工厂类&#xff08;Java代码&#xff09;&#xff0c;定义实例工厂&#xff08;xml&#xff09;&#xff0c;定义调用实例工厂&#xff…...

今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存

文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...

LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf

FTP 客服管理系统 实现kefu123登录&#xff0c;不允许匿名访问&#xff0c;kefu只能访问/data/kefu目录&#xff0c;不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...

[大语言模型]在个人电脑上部署ollama 并进行管理,最后配置AI程序开发助手.

ollama官网: 下载 https://ollama.com/ 安装 查看可以使用的模型 https://ollama.com/search 例如 https://ollama.com/library/deepseek-r1/tags # deepseek-r1:7bollama pull deepseek-r1:7b改token数量为409622 16384 ollama命令说明 ollama serve #&#xff1a…...

如何配置一个sql server使得其它用户可以通过excel odbc获取数据

要让其他用户通过 Excel 使用 ODBC 连接到 SQL Server 获取数据&#xff0c;你需要完成以下配置步骤&#xff1a; ✅ 一、在 SQL Server 端配置&#xff08;服务器设置&#xff09; 1. 启用 TCP/IP 协议 打开 “SQL Server 配置管理器”。导航到&#xff1a;SQL Server 网络配…...

Axure 下拉框联动

实现选省、选完省之后选对应省份下的市区...

如何在Windows本机安装Python并确保与Python.NET兼容

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…...

[拓扑优化] 1.概述

常见的拓扑优化方法有&#xff1a;均匀化法、变密度法、渐进结构优化法、水平集法、移动可变形组件法等。 常见的数值计算方法有&#xff1a;有限元法、有限差分法、边界元法、离散元法、无网格法、扩展有限元法、等几何分析等。 将上述数值计算方法与拓扑优化方法结合&#…...