使用Pygame制作“打砖块”游戏

devtools/2025/2/2 17:48:22/

1. 前言

打砖块(Breakout / Arkanoid) 是一款经典街机游戏,玩家控制一个可左右移动的挡板,接住并反弹球,击碎屏幕上方的砖块。随着砖块被击碎,不仅能获得分数,还可以体验到不断加速或复杂的反弹乐趣。 在本篇文章里,我们将使用 Python 3.x + Pygame 库,手把手实现一个简易版本的打砖块游戏,包含最核心的移动、碰撞和得分功能。


2. 开发环境与准备

  1. Python 3.x
  2. Pygame:若尚未安装,可通过命令 pip install pygame 进行安装。
  3. 桌面系统:Windows、macOS 或绝大多数 Linux 桌面环境都能正常使用 Pygame。

安装完成后,使用 import pygame 测试是否成功即可。


3. 游戏思路

要完成一个打砖块游戏,需要实现以下几个关键模块:

  1. 挡板(Paddle)

    • 位于屏幕底部,可左右移动。
    • 通过键盘或鼠标控制位置。
  2. 球(Ball)

    • 从挡板上方出发,向上运动;
    • 在碰到墙壁时发生反弹;
    • 在碰到挡板或砖块时,需要计算反弹方向,并可能击碎砖块、加分。
  3. 砖块(Bricks)

    • 通常在屏幕上方排列成若干行;
    • 一旦被球击中,会被击碎并增加分数;
    • 也可以设定一些特殊砖块,击中后会产生道具等(此处仅做简易实现)。
  4. 游戏结束

    • 若球掉出屏幕底部则表示丢失一条命,或者直接游戏结束;
    • 如果所有砖块都被击碎,则玩家胜利。

4. 完整示例代码

将以下示例保存为 breakout_game.py 并运行,即可体验一个最基本的打砖块游戏。你也可以根据需求自由添加更多功能或美化界面。

python">import pygame
import sys
import random# 初始化 Pygame
pygame.init()# ----------------------
#  全局配置
# ----------------------
WIDTH, HEIGHT = 600, 600   # 游戏窗口大小
FPS = 60                   # 帧率# 颜色
BLACK = (0, 0, 0)
WHITE = (255, 255, 255)
GRAY  = (100, 100, 100)
RED   = (255, 0, 0)
GREEN = (0, 255, 0)
BLUE  = (0, 0, 255)
YELLOW = (255, 255, 0)# 游戏窗口
screen = pygame.display.set_mode((WIDTH, HEIGHT))
pygame.display.set_caption("打砖块 - Pygame 示例")
clock = pygame.time.Clock()# 字体
font = pygame.font.SysFont("arial", 24)# ----------------------
#  挡板类
# ----------------------
class Paddle:def __init__(self):self.width = 100self.height = 15self.x = (WIDTH - self.width) // 2self.y = HEIGHT - 50self.speed = 8def draw(self, surface):pygame.draw.rect(surface, BLUE, (self.x, self.y, self.width, self.height))def move_left(self):self.x -= self.speedif self.x < 0:self.x = 0def move_right(self):self.x += self.speedif self.x + self.width > WIDTH:self.x = WIDTH - self.width# ----------------------
#  球类
# ----------------------
class Ball:def __init__(self, paddle):self.radius = 8self.x = paddle.x + paddle.width // 2self.y = paddle.y - 10# 球初始速度self.speed_x = random.choice([-4, 4])self.speed_y = -4def draw(self, surface):pygame.draw.circle(surface, RED, (int(self.x), int(self.y)), self.radius)def update(self, paddle, bricks):"""更新球的位置、检查墙体碰撞、挡板碰撞和砖块碰撞"""self.x += self.speed_xself.y += self.speed_y# 碰撞左右墙if self.x - self.radius < 0:self.x = self.radiusself.speed_x = -self.speed_xelif self.x + self.radius > WIDTH:self.x = WIDTH - self.radiusself.speed_x = -self.speed_x# 碰撞上墙if self.y - self.radius < 0:self.y = self.radiusself.speed_y = -self.speed_y# 掉到底部 -> 游戏结束的处理可在主循环中判断# if self.y + self.radius > HEIGHT:#    pass# 碰撞挡板if (self.x > paddle.x and self.x < paddle.x + paddle.widthand self.y + self.radius > paddle.yand self.y - self.radius < paddle.y + paddle.height):self.speed_y = -self.speed_y# 球可能根据撞击位置微调水平速度(可选)# self.speed_x += random.choice([-1, 0, 1])# 碰撞砖块for brick in bricks[:]:if (self.x + self.radius > brick.xand self.x - self.radius < brick.x + brick.widthand self.y + self.radius > brick.yand self.y - self.radius < brick.y + brick.height):bricks.remove(brick)self.speed_y = -self.speed_y  # 简化反弹,只改变垂直方向return 10  # 得分10return 0# ----------------------
#  砖块类
# ----------------------
class Brick:def __init__(self, x, y, width, height, color=GREEN):self.x = xself.y = yself.width = widthself.height = heightself.color = colordef draw(self, surface):pygame.draw.rect(surface, self.color, (self.x, self.y, self.width, self.height))pygame.draw.rect(surface, BLACK, (self.x, self.y, self.width, self.height), 1)  # 边框def create_bricks(rows=5, cols=8):"""创建指定行列的砖块,返回列表"""bricks = []brick_width = (WIDTH - 40) // colsbrick_height = 20x_offset = 20y_offset = 40for row in range(rows):for col in range(cols):x = x_offset + col * brick_widthy = y_offset + row * brick_heightcolor = random.choice([GREEN, YELLOW, GRAY, BLUE])brick = Brick(x, y, brick_width, brick_height, color)bricks.append(brick)return bricks# ----------------------
#  主函数
# ----------------------
def main():paddle = Paddle()ball = Ball(paddle)bricks = create_bricks(rows=5, cols=8)score = 0running = Truewhile running:clock.tick(FPS)for event in pygame.event.get():if event.type == pygame.QUIT:running = False# 键盘输入keys = pygame.key.get_pressed()if keys[pygame.K_LEFT]:paddle.move_left()if keys[pygame.K_RIGHT]:paddle.move_right()# 更新球的位置gained_score = ball.update(paddle, bricks)score += gained_score# 判断游戏结束:如果球掉到底部 or 砖块全部消失if ball.y - ball.radius > HEIGHT:# 球掉出底部running = Falseif len(bricks) == 0:# 所有砖块被击破running = False# 绘制screen.fill(BLACK)# 画砖块for brick in bricks:brick.draw(screen)# 画挡板和球paddle.draw(screen)ball.draw(screen)# 显示分数text_surface = font.render(f"Score: {score}", True, WHITE)screen.blit(text_surface, (10, 10))pygame.display.flip()# 游戏结束后显示结果game_over(score)def game_over(score):"""游戏结束界面"""screen.fill(GRAY)msg = f"Game Over! Your Score: {score}"label = font.render(msg, True, BLACK)# 居中显示rect = label.get_rect(center=(WIDTH // 2, HEIGHT // 2))screen.blit(label, rect)pygame.display.flip()pygame.time.wait(3000)pygame.quit()sys.exit()if __name__ == "__main__":main()

核心逻辑解读

  1. Paddle(挡板)

    • 只需实现水平移动;
    • 在超出屏幕边界时,强制回到合法范围内。
  2. Ball(球)

    • 通过 self.x, self.y 表示球心位置,self.speed_x, self.speed_y 表示当前水平和垂直速度;
    • 每帧更新时,先加上速度;遇到左右墙、上墙时反转速度;遇到底部则表示掉落。
    • 在与挡板或砖块相交时,需要根据碰撞方向做出相应反弹。
  3. Brick(砖块)

    • 仅保存坐标、宽高、颜色;
    • 被击中后从列表中移除。
  4. 碰撞检测

    • 简化实现:只要球的圆心与砖块矩形区域重叠即可判断为碰撞;
    • 在真实游戏中,可以做更精确的检测(圆与矩形边的距离、角度等)或更细致的物理反弹。
  5. 游戏结束

    • 玩家失败:球掉出屏幕底部;
    • 玩家胜利:所有砖块被清除。

5. 实现效果

image.png

image.png


6. 总结

通过本篇文章,你已经学会了如何使用 Python + Pygame 从零构建一个基础的打砖块游戏。该示例涵盖了碰撞检测游戏循环对象管理等常见2D游戏开发中的核心逻辑。你可以在此基础上自由发挥,加入更多道具、特效和关卡,从而打造一个更完整、更丰富的打砖块游戏


http://www.ppmy.cn/devtools/155499.html

相关文章

C++ 写一个简单的加减法计算器

************* C topic&#xff1a;结构 ************* Structure is a very intersting issue. I really dont like concepts as it is boring. I would like to cases instead. If I want to learn something, donot hesitate to make shits. Like building a house. Wh…

http跳转https

1、第一种&#xff1a;不好使 在nginx的配置中&#xff0c;在https的server站点添加如下头部&#xff1a; add_header Strict-Transport-Security “max-age63072000; includeSubdomains; preload”; 这样当第一次以https方式访问我的网站&#xff0c;nginx则会告知客户端的浏览…

详解u3d之AssetBundle

一.AssetBundle的概念 “AssetBundle”可以指两种不同但相关的东西。 1.1 AssetBundle指的是u3d在磁盘上生成的存放资源的目录 目录包含两种类型文件(下文简称AB包)&#xff1a; 一个序列化文件&#xff0c;其中包含分解为各个对象并写入此单个文件的资源。资源文件&#x…

HTML 标题

HTML 标题 引言 HTML&#xff08;超文本标记语言&#xff09;是构建网页的基础&#xff0c;而标题则是网页中不可或缺的元素。标题不仅能够帮助用户快速了解网页内容&#xff0c;还能够对搜索引擎优化&#xff08;SEO&#xff09;产生重要影响。本文将详细介绍HTML标题的用法…

图书管理系统 Axios 源码__新增图书

目录 功能介绍 核心代码解析 源码&#xff1a;新增图书功能 总结 本项目基于 HTML、Bootstrap、JavaScript 和 Axios 开发&#xff0c;实现了图书的增删改查功能。以下是新增图书的功能实现&#xff0c;适合前端开发学习和项目实践。 功能介绍 用户可以通过 模态框&#xf…

项目测试之MockMvc

文章目录 基础基础概念Mockxxx一般实现文件位置 实战MockMvc与Test注解不兼容RequestParams参数RequestBody参数 基础 基础概念 定义&#xff1a;是Spring框架提供的一种用于测试Spring MVC控制器的工具&#xff0c;它允许开发者在不启动完整的web服务器的情况下&#xff0c;…

为AI聊天工具添加一个知识系统 之82 详细设计之23 符号逻辑 正则表达式规则 之1

本文要点 在继续“逻辑符号”的设计中&#xff0c;我们先回顾一个本项目--作为备忘也作为 设计中的时刻牢记&#xff1a; 回顾 项目介绍 项目名&#xff1a;为使用AI聊天工具的聊天者开挂一个知识系统项目口号&#xff1a;通过不断完善一个概念整体运营的大局观思想来持续维…

C++编程语言:抽象机制:模板(Bjarne Stroustrup)

目录 23.1 引言和概观(Introduction and Overview) 23.2 一个简单的字符串模板(A Simple String Template) 23.2.1 模板的定义(Defining a Template) 23.2.2 模板实例化(Template Instantiation) 23.3 类型检查(Type Checking) 23.3.1 类型等价(Type Equivalence) …