python小游戏代码大全-Python小游戏之300行代码实现俄罗斯方块

article/2025/8/8 16:50:43

前言

本文代码基于 python3.6 和 pygame1.9.4。

俄罗斯方块是儿时最经典的游戏之一,刚开始接触 pygame 的时候就想写一个俄罗斯方块。但是想到旋转,停靠,消除等操作,感觉好像很难啊,等真正写完了发现,一共也就 300 行代码,并没有什么难的。

先来看一个游戏截图,有点丑,好吧,我没啥美术细胞,但是主体功能都实现了,可以玩起来。

201901041045181.png

现在来看一下实现的过程。

外形

俄罗斯方块整个界面分为两部分,一部分是左边的游戏区域,另一部分是右边的显示区域,显示得分、速度、下一个方块样式等。这里就不放截图了,看上图就可以。

游戏区域跟贪吃蛇一样,是由一个个小方格组成的,为了看得直观,我特意画了网格线。

import sys

import pygame

from pygame.locals import *

SIZE = 30 # 每个小方格大小

BLOCK_HEIGHT = 20 # 游戏区高度

BLOCK_WIDTH = 10 # 游戏区宽度

BORDER_WIDTH = 4 # 游戏区边框宽度

BORDER_COLOR = (40, 40, 200) # 游戏区边框颜色

SCREEN_WIDTH = SIZE * (BLOCK_WIDTH + 5) # 游戏屏幕的宽

SCREEN_HEIGHT = SIZE * BLOCK_HEIGHT # 游戏屏幕的高

BG_COLOR = (40, 40, 60) # 背景色

BLACK = (0, 0, 0)

def print_text(screen, font, x, y, text, fcolor=(255, 255, 255)):

imgText = font.render(text, True, fcolor)

screen.blit(imgText, (x, y))

def main():

pygame.init()

screen = pygame.display.set_mode((SCREEN_WIDTH, SCREEN_HEIGHT))

pygame.display.set_caption('俄罗斯方块')

font1 = pygame.font.SysFont('SimHei', 24) # 黑体24

font_pos_x = BLOCK_WIDTH * SIZE + BORDER_WIDTH + 10 # 右侧信息显示区域字体位置的X坐标

font1_height = int(font1.size('得分')[1])

score = 0 # 得分

while True:

for event in pygame.event.get():

if event.type == QUIT:

sys.exit()

# 填充背景色

screen.fill(BG_COLOR)

# 画游戏区域分隔线

pygame.draw.line(screen, BORDER_COLOR,

(SIZE * BLOCK_WIDTH + BORDER_WIDTH // 2, 0),

(SIZE * BLOCK_WIDTH + BORDER_WIDTH // 2, SCREEN_HEIGHT), BORDER_WIDTH)

# 画网格线 竖线

for x in range(BLOCK_WIDTH):

pygame.draw.line(screen, BLACK, (x * SIZE, 0), (x * SIZE, SCREEN_HEIGHT), 1)

# 画网格线 横线

for y in range(BLOCK_HEIGHT):

pygame.draw.line(screen, BLACK, (0, y * SIZE), (BLOCK_WIDTH * SIZE, y * SIZE), 1)

print_text(screen, font1, font_pos_x, 10, f'得分: ')

print_text(screen, font1, font_pos_x, 10 + font1_height + 6, f'{score}')

print_text(screen, font1, font_pos_x, 20 + (font1_height + 6) * 2, f'速度: ')

print_text(screen, font1, font_pos_x, 20 + (font1_height + 6) * 3, f'{score // 10000}')

print_text(screen, font1, font_pos_x, 30 + (font1_height + 6) * 4, f'下一个:')

pygame.display.flip()

if __name__ == '__main__':

main()

方块

接下来就是要定义方块,方块的形状一共有以下 7 种:

201901041045182.png

I 型

201901041045183.png

O 型

201901041045184.png

T 型

201901041045195.png

S 型

201901041045196.png

Z 型

201901041045197.png

L 型

201901041045198.png

J 型

这里我做了多次的更改,因为方块最大的长度是长条形的,为4格,所以我统一用了 4 × 4 的方格来定义。这也是可以的,只是后来发现不方便。

为了直观,直接以一个二维数组来定义方块,其中 . 表示空的, 0 表示实心的。(用 . 表示空是为了看得直观,如果用空格会看不清。)

例如 I 行,以 4 × 4 方格定义为

['.0..',

'.0..',

'.0..',

'.0..']

['....',

'....',

'0000',

'....']

方块最难的是需要实现旋转功能,比如 I 型,就有横和竖两种形态。所谓旋转,表面上看,是把方块顺时针旋转了 90°,但实际做的时候,我们并不需要正真的去实现这个“旋转”的效果。

最终实现的时候,这些图形都是我们画在界面上的,而每一次刷新,界面上所有内容都会被清空重画,所以旋转只是画当前方块的时候不再画之前的形状,而是画旋转后的形状。

比如这个 I 型,定义成了 4 × 4 的形状,但实际上只需要 1 × 4 或 4 × 1 就可以了,其他剩下的地方都是空的。它不像 T 型,T 型不是一个矩形,如果用一个矩形来定义,必然有 2 个位置是空的。那么,I 型真的有必要定义成 4 × 4 吗?

答案是肯定的。想想看,如果是 4 × 1 的一个横条,旋转后变成 1 × 4 的竖条,这个位置怎么确定?好像有点困难。但是如果是 4 × 4 的正方形,我们只需要固定起点坐标(左上角)不变,把竖条的 4 × 4 直接替换掉横条的 4 × 4 区域,是不是就实现旋转了?而且位置很容易计算。

另外一点,在有些情况下是不可以旋转的。比如 I 型的竖条,在紧贴左右边框的时候是不可以旋转的。这点我有印象,可以肯定。但是对于其他的形状,我就不是很确定了,我百度搜了下,找了个网页版的俄罗斯方块玩了下,发现也是不可以的。例如:

201901041045199.png

在紧贴右边框的时候是无法旋转的。如果要每一个形状都去判断一下,那实在是太烦了。从方块的定义入手,就可以很简单的实现。

例如竖条行,定义是:

['.0..',

'.0..',

'.0..',

'.0..']

竖条是可以贴边的,所以当它在最左边的时候,X 轴坐标是 -1,这是因为定义中左边一竖排是空的。我们只需判定,当方块所定义的形状(包括空的部分)完全在游戏区域内时才可以旋转。

我之前所说,全都定义成 4 × 4 不好,原因就在这里,对于 T 型等其他形状,无法做这个判定。所以,对于 T 型等形状,我们可以定义成 3 × 3 的格式:

['.0.',

'000',

'...']

还有一种情况是无法旋转的,就是旋转后的位置已经被别的方块占了。另外下落,左右移动,都要做这个判断。既然这些是一致的,那么就可以用同一个方法来判断。

先要定义一个 game_area 变量,用于存放整个游戏区域当前的状态:

game_area = [['.'] * BLOCK_WIDTH for _ in range(BLOCK_HEIGHT)]

初始状态全是空的,所以全部用 . 初始化就可以了。

另外,需要一些变量定义当前下落方块的状态

cur_block = None # 当前下落方块

cur_pos_x, cur_pos_y = 0, 0 # 当前下落方块的坐标

方块我们是以二维数组的方式定义的,并且存在空行和空列,如果我们遍历这个二维数组判断其所在的区域在当前游戏区域内是否已经被别的方块所占,这个是可以实现的。我们考虑另外一种情况,一个竖条形,左边一排是空的,这空的一排是可以移出游戏区域的,这个怎么判断?每次左移的时候都去判断一下左边一排全都是空吗?这太麻烦了。并且方块都是固定的,所以这些我们可以提前定义好。最终方块定义如下:

from collections import namedtuple

Point = namedtuple('Point', 'X Y')

Block = namedtuple('Block', 'template start_pos end_pos name next')

# S形方块

S_BLOCK = [Block(['.00',

'00.',

'...'], Point(0, 0), Point(2, 1), 'S', 1),

Block(['0..',

'00.',

'.0.'], Point(0, 0), Point(1, 2), 'S', 0)]

方块需要包含两个方法,获取随机一个方块和旋转时获取旋转后的方块

BLOCKS = {'O': O_BLOCK,

'I': I_BLOCK,

'Z': Z_BLOCK,

'T': T_BLOCK,

'L': L_BLOCK,

'S': S_BLOCK,

'J': J_BLOCK}

def get_block():

block_name = random.choice('OIZTLSJ')

b = BLOCKS[block_name]

idx = random.randint(0, len(b) - 1)

return b[idx]

# 获取旋转后的方块

def get_next_block(block):

b = BLOCKS[block.name]

return b[block.next]

判断是否可以旋转,下落,移动的方法也很容易实现了

def _judge(pos_x, pos_y, block):

nonlocal game_area

for _i in range(block.start_pos.Y, block.end_pos.Y + 1):

if pos_y + block.end_pos.Y >= BLOCK_HEIGHT:

return False

for _j in range(block.start_pos.X, block.end_pos.X + 1):

if pos_y + _i >= 0 and block.template[_i][_j] != '.' and game_area[pos_y + _i][pos_x + _j] != '.':

return False

return True

停靠

最后一个问题是停靠,当方块下落到底或者遇到别的方块之后,就不能在下落了。我将此称之为“停靠”,有个名字说起来也方便一点。

首先是要判断是否可以停靠,停靠发生之后,就是将当前方块的非空点画到游戏区域上,说白了,就是将cur_block的非空点按对应位置复制到game_area里去。并且计算是否有一排被全部填满了,全部填满则消除。

def _dock():

nonlocal cur_block, next_block, game_area, cur_pos_x, cur_pos_y, game_over

for _i in range(cur_block.start_pos.Y, cur_block.end_pos.Y + 1):

for _j in range(cur_block.start_pos.X, cur_block.end_pos.X + 1):

if cur_block.template[_i][_j] != '.':

game_area[cur_pos_y + _i][cur_pos_x + _j] = '0'

if cur_pos_y + cur_block.start_pos.Y <= 0:

game_over = True

else:

# 计算消除

remove_idxs = []

for _i in range(cur_block.start_pos.Y, cur_block.end_pos.Y + 1):

if all(_x == '0' for _x in game_area[cur_pos_y + _i]):

remove_idxs.append(cur_pos_y + _i)

if remove_idxs:

# 消除

_i = _j = remove_idxs[-1]

while _i >= 0:

while _j in remove_idxs:

_j -= 1

if _j < 0:

game_area[_i] = ['.'] * BLOCK_WIDTH

else:

game_area[_i] = game_area[_j]

_i -= 1

_j -= 1

cur_block = next_block

next_block = blocks.get_block()

cur_pos_x, cur_pos_y = (BLOCK_WIDTH - cur_block.end_pos.X - 1) // 2, -1 - cur_block.end_pos.Y

至此,整个俄罗斯方块的主体功能就算是完成了。

这里很多参数是可以调的,例如觉得旋转别扭,可以直接调整方块的定义,而无需去改动代码逻辑。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。


http://chatgpt.dhexx.cn/article/lS2PH6nI.shtml

相关文章

python编程游戏代码大全,python简单的小游戏代码

大家好&#xff0c;本文将围绕python编程一个最简单游戏代码展开说明&#xff0c;20行python代码的入门级小游戏是一个很多人都想弄明白的事情&#xff0c;想搞清楚python游戏编程入门游戏代码需要先了解以下几个事情。 一、石头剪刀布游戏 目标&#xff1a;创建一个命令行游戏…

干货来啦!!!二十种Python代码游戏源代码分享

学Python中&#xff0c;自我感觉学的还不错的亚子~想做点什么来练练手&#xff0c;然后我疯狂的找各种小游戏的教程源码什么的&#xff0c;于是我就疯狂的找呀找呀&#xff0c;就找到了一大堆&#xff0c;哈哈哈 毕竟我是从小就有一个游戏梦&#xff0c;现在就弥补一下自己小时…

python小游戏代码大全-python小游戏实现代码

早上逛CSDN首页就见到这么一篇教程。看了一下很有意思,就马上动手实现了一下。看看效果吧: 完整代码: # -*- coding: utf-8 -*- # 1 - Import library import pygame from pygame.locals import * import math import random # 2 - Initialize the game keys = [Fal…

python一行代码制作20款经典游戏

今天分享一个有趣的Python游戏库freegames&#xff0c;它包含20余款经典小游戏&#xff0c;像贪吃蛇、吃豆人、乒乓、数字华容道等等&#xff0c;依托于标准库Turtle。 我们不仅可以通过1行代码进行重温这些童年小游戏&#xff0c;还可以查看源码自己学习下游戏编写&#xff0…

python简单小游戏代码教程,python小游戏程序源代码

球球各位大神怎么用python写一个猜词小游戏的代码&#xff1f; key input(请输入一个单词&#xff1a;)description input(输入单词描述&#xff1a;)chance 5mark 5print(现在开始游戏)print(description \t 这是单词的描述,请你输入这个单词&#xff1a; )for i in ra…

Python代码,能玩30多款童年游戏!这些有几个是你玩过的

大游戏小游戏有千千万万&#xff0c;这些小游戏应该只有90后才玩过和懂吧 儿童节即将到来&#xff0c;虽然秃头程序员没有头发&#xff0c;但是童心还是一直都在的&#xff0c;今天就分享一些私藏的童年游戏&#xff0c;十几行代码就能进入使用Python开发的小游戏快乐玩耍&…

一口气用Python写了13个小游戏(附源码)

仅限技术学习参考 分享13个游戏源码&#xff0c;可以自己复现玩玩&#xff0c;研究下里面的编程逻辑&#xff0c;对学习编程&#xff08;特别是初学者&#xff09;应该会有很大帮助。 1、吃金币 源码分享&#xff1a; import os import cfg import sys import pygame import…

JSP校验必填项

输入框为input标签&#xff0c;在保存的function() 里增加校验&#xff1b;

android textview 必填,在android中如何使用Html渲染的方式实现必填项前面的*号

本篇文章主要介绍了android中使用Html渲染的方式实现必填项前面的*号示例&#xff0c;具有一定的参考价值&#xff0c;有兴趣的可以了解一下 项目的个人基本信息UI界面效果图如下&#xff0c;有一个红色的*号&#xff0c;并且跟它挨着的文字颜色不一样。简友们&#xff0c;你们…

php邮箱必填,WordPress移除用户新注册时邮件必填选项(注册不填邮箱/邮箱非必填)...

使用WordPress搭建的网站是支持用户注册做商城网站的&#xff0c;注册的时候默认要求用户填写一个邮箱&#xff0c;并且是必须填写的&#xff0c;而某些网站情况特殊&#xff0c;可能并不需要强制填写邮件&#xff0c;所以我们可以通过下面的代码把强制填写邮件功能改为必填项目…

element ui表单必填_详解element-ui设置下拉选择切换必填和非必填

? 需求 默认都是必选 下拉选择的时候 选择必填&#xff0c;活动名称为必填&#xff0c;需要校验和显示* 选择非必填&#xff0c;活动名称不做校验&#xff0c;隐藏* ? 初始校验规则 经测试&#xff0c;网上其他的方式都没有实现需求&#xff0c;动态切换rules中的required没有…

必填校验 验证问题

今天做项目的时候&#xff0c;突然发现在新增和保存的时候有个必填校验明明加了&#xff0c;但是没有填写&#xff0c;结果竟然没有提示。原来是接收的验证对象不一样造成的&#xff0c;从而忽略了对他的校验。 当我对处理应收单类型没选取的时候&#xff0c;进行新增或者保存…

easypoi必填项_easypoi必填项_EasyPoi使用入门

咱们在开发的时候&#xff0c;总会遇到需要通过代码操作办公软件的情况&#xff0c;而excel与word的操作最为频繁。 当然我们Java程序员可以选择JXL或者POI来完成相应的Excel操作&#xff0c;但是大家用过都知道&#xff0c;有些地方感觉还是不够简单&#xff0c;不那么尽如人意…

php邮箱必填,discuz关闭邮箱注册必填选项

由于需要,我要做的网站不需要注册的时候填写邮件,需要关闭必填,于是从网上找了教程,发现没有后台的选项,之后知道,discuz3.2之后就取消 取消邮箱必填 这个选项了,于是从网上找了点资料,成功关闭邮箱注册必填选项。 情况1:disczu3.2之前的版本,直接在后台-----全局--…

html 必填设置,html如何设置必填项

在html中&#xff0c;可以使用required属性来设置必填项&#xff0c;需要在input元素标签中添加“required"required"”样式即可。required属性规定必需在提交之前填写输入字段。如果使用该属性&#xff0c;则字段是必填(或必选)的。 本教程操作环境&#xff1a;wind…

vue 设置表单必填项

1.要求 在做一些用户信息相关的功能时&#xff0c;经常用到表单项去收集数据&#xff0c;其中有些属性必须填写&#xff0c;要求如图&#xff1a; 2.实现方法 在data中添加一个rules来规定&#xff1a; rules: {no: [{required: true, message: 请输入账号, trigger:…

vue设置必填项和判断必填项是否填入的弹窗提示

项目场景&#xff1a; 有的功能需要设置必填项,当然也需要判断是不是添上了,还需要加上提示框 问题描述 1.如何设置必填项 2.如何设置弹窗提示 3.如何将二者结合起来实现点击时既可以判断必填项也可以弹出提示框 原因分析&#xff1a; 其他两个我没有什么问题,在将二者结合…

莫队算法——从入门到黑题

众所周知,莫队是由莫涛大神提出的,一种玄学毒瘤暴力骗分区间操作算法,它以简短的框架、简单易记的板子和优秀的复杂度闻名于世。然而由于莫队算法应用的毒瘤,很多可做的莫队模板题都有着较高的难度评级,令很多初学者望而却步。然而,如果你真正理解了莫队的算法原理,那么…

莫队(模板 + 带简单修改的莫队)

莫队 这个算法的思想比较简单&#xff0c;我们在做RMQ类问题时&#xff0c;有多次询问的那种&#xff0c;其实在这些询问中有很多都是问的同一段区间&#xff0c;即有的区间被询问的多次&#xff0c;所以我们对询问进行一个排序&#xff0c;假设上次询问我们得到了区间[l,r]的…

莫队 - 基础与扩展

普通莫队 莫队可以说是一个算法&#xff0c;但更多是一种思想。 我们先来看看普通莫队解决的问题&#xff1a; 有一个长度为 n n n 的数列 a a a。 q q q 个询问&#xff1a; a a a 在 [ l i , r i ] [{l_i},r_i] [li​,ri​] 中有多少个不同的数。 不强制在线。 1 ≤ n …