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

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

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

image

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

外形

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

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

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 种:

image

I 型

image

O 型

image

T 型

image

S 型

image

Z 型

image

L 型

image

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 型的竖条,在紧贴左右边框的时候是不可以旋转的。这点我有印象,可以肯定。但是对于其他的形状,我就不是很确定了,我百度搜了下,找了个网页版的俄罗斯方块玩了下,发现也是不可以的。例如:

image

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

例如竖条行,定义是:

['.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`

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

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

关注+转发然后私信回复 “俄罗斯方块” 获取源码。

python编写游戏300行代码_300行代码实现Python游戏:俄罗斯方块相关推荐

  1. 用Python编写干净 可测试 高质量的代码

    用Python编写干净 可测试 高质量的代码 您不必听信我的观点,也不必直接研究代码,可以通过其他方法明显地看出这两种风格之间的差异.第一种方法是实际度量得到测试的代码行数.Nose 是一种流行的 P ...

  2. 使用python编写从抖音下载视频的代码

    使用 Python 编写从抖音下载视频的代码,可以使用第三方库来帮助你完成这个任务. 其中一种方法是使用 PyDouyin 库,这是一个 Python 库,可以帮助你从抖音下载视频. 要使用 PyDo ...

  3. python编写用户输入的是q么代码_Python课 #01号作业

    为了记录我的Python课,将我的作业发上来,欢迎各位大佬评鉴.如果你有什么更好的想法,请在下方评论或联系我.谢谢! 作业一:向某人打招呼 描述 程序接收用户输入的姓名,然后输出向该姓名问好的文字. ...

  4. 找不到libmmd.dll无法继续执行代码_300 行代码带你秒懂 Java 多线程!| 原力计划...

    作者 | 永远在路上[] 责编 | 胡巍巍 出品 | CSDN博客 线程线程的概念,百度是这样解释的:线程(英语:Thread)是操作系统能够进行运算调度的最小单位.它被包含在进程之中,是进程中的实际 ...

  5. 用python编写一个求偶数阶乘的函数_一行Python代码写阶乘函数

    原标题:一行Python代码写阶乘函数 背景 我以前用Dart写过一行的阶乘函数: fact(double n) => n < 0 ? throw ('n must be bigger t ...

  6. python编写抢座位软件_程序员硬核Python抢票教程”,帮你抢回家车票

    盼望着,盼望着,春节的脚步近了,然而,每年到这个时候,最难的,莫过于一张回家的火车票. 据悉,今年春运期间,全国铁路发送旅客人次同比将增长8.0%,达到4.4亿人次,2020年铁路春运自1月10日开始 ...

  7. 用python编写缠论中枢_【缠论】python 实现

    缠论来源于缠中说缠博客,欢迎加微信探讨,我的微信号是 zengbin93 考虑到我自己对缠论的理解还在"四不像"水平,czsc 这个库只有参考的价值.画画笔和线段也还行,但这个离缠 ...

  8. python编写程序输出诗句_闲来无事能干嘛 用Python来玩诗歌接龙

    闲来无事能干嘛 用Python来玩诗歌接龙 作为一个懂Python爬虫的运维狗,闲来无事的时候总要找点乐子(睡觉不香么),哈哈,就是这么的敬业(其实是无聊).今天网盾科技给大家讲讲怎么用Python爬 ...

  9. python编写递归函数_Python递归函数如何写?正确的Python递归函数用法!

    在函数内部,可以调用其他函数.如果一个函数在内部调用自身本身,这个函数就是递归函数. 一.举个例子,我们来计算阶乘n! = 1 x 2 x 3 x - x n,用函数fact(n)表示,可以看出: f ...

最新文章

  1. ConcurrentHashMap源码分析(1)——JDK1.7的实现
  2. datagirdview跟据内容自动适应单元格大小
  3. matlab 比较日期,[转载][Matlab]关于时间的函数的不完全总结
  4. 这四种攻击单片机的主要技术你了解多少?
  5. libevent: linux安装libevent
  6. Bzoj3065 带插入区间K小值
  7. 拓端tecdat|R语言资产配置: 季度战术资产配置策略研究
  8. Mat和IplImage
  9. 史上最全的贝塞尔曲线(Bezier)全解(三):贝塞尔曲线实现满屏爱心
  10. Codeforces 938E Max History [排列组合]
  11. python 定时程序_python每天定时运行某程序代码
  12. 微信小程序自定义地址选择器
  13. java一个方法排他调用_Java编程实现排他锁代码详解
  14. C语言(二):数据类型
  15. word2013无法打开doc文件显示正在受保护视图中打开解决方法
  16. Oracle分区表的使用
  17. 大一学生网页课程作业 南京介绍网页设计 学生家乡网页设计作品静态 HTML网页模板源码 html我的家乡网页作业
  18. 使用markdown玩转代码架构设计
  19. Linux运维交流学习
  20. 少儿python编程课程大纲_1.Python编程-课程教学大纲.doc

热门文章

  1. Access函数大全
  2. 解决Navicat for MySQL 1045错误的三种方法
  3. 极客日报第 30 期:Google回应全球宕机:磁盘满了;摩拜App昨晚正式停止服务
  4. 服务器能买断嘛?服务器租用多少钱一个月呢?
  5. SSH 协议的 ssh StrictHostKeyChecking
  6. html 还原键盘默认事件,键盘肿么还原为默认键位
  7. Java解析excel操作
  8. php验证qq,php验证QQ邮箱是否有效
  9. 【《现代操作系统 第4版》】4、进程间的通信之互斥
  10. 旧电脑改路由器加文件服务器,如何将旧电脑改成软路由器