python实现俄罗斯方块游戏


Posted in Python onMarch 25, 2020

在公司实习。公司推崇Python和Django框架,所以也得跟着学点。

简单瞅了下Tkinter,和Canvas配合在一起,还算是简洁的界面开发API。threading.Thread创建新的线程,其多线程机制也算是方便。

只是canvas.create_rectangle居然不是绘制矩形,而是新建了矩形控件这点让人大跌眼镜。先开始,在线程里每次都重绘多个矩形(随数组变化),其实是每次都新建了N个矩形,结果内存暴增。原来,对矩形进行变更时,只需用canvas.itemconfig即可。

下面就是截图(时间太晚,明日还得上班,做得非常粗糙...没事时再慢慢修正)。

python实现俄罗斯方块游戏

而代码如下:

#coding=utf-8 
from Tkinter import *; 
from random import *; 
import thread; 
from tkMessageBox import showinfo; 
import threading; 
from time import sleep; 
class BrickGame(object): 
 
 #是否开始 
 start = True; 
 #是否到达底部 
 isDown = True; 
 
 #窗体 
 window = None; 
 #frame 
 frame1 = None; 
 
 #绘图类 
 canvas = None; 
 
 #标题 
 title = "BrickGame"; 
 #宽和高 
 width = 350; 
 height = 670; 
 
 #行和列 
 rows = 20; 
 cols = 10; 
 
 #几种方块 
 brick = [ 
 
 [ 
 [ 
 [1,1,1], 
 [0,0,1], 
 [0,0,0] 
 ], 
 [ 
 [0,0,1], 
 [0,0,1], 
 [0,1,1] 
 ], 
 [ 
 [0,0,0], 
 [1,0,0], 
 [1,1,1] 
 ], 
 [ 
 [1,1,0], 
 [1,0,0], 
 [1,0,0] 
 ] 
 ], 
 [ 
 [ 
 [0,0,0], 
 [0,1,1], 
 [0,1,1] 
 ], 
 [ 
 [0,0,0], 
 [0,1,1], 
 [0,1,1] 
 ], 
 [ 
 [0,0,0], 
 [0,1,1], 
 [0,1,1] 
 ], 
 [ 
 [0,0,0], 
 [0,1,1], 
 [0,1,1] 
 ] 
 ], 
 [ 
 [ 
 [1,1,1], 
 [0,1,0], 
 [0,1,0] 
 ], 
 [ 
 [0,0,1], 
 [1,1,1], 
 [0,0,1] 
 ], 
 [ 
 [0,1,0], 
 [0,1,0], 
 [1,1,1] 
 ], 
 [ 
 [1,0,0], 
 [1,1,1], 
 [1,0,0] 
 ] 
 ], 
 [ 
 [ 
 [0,1,0], 
 [0,1,0], 
 [0,1,0] 
 ], 
 [ 
 [0,0,0], 
 [1,1,1], 
 [0,0,0] 
 ], 
 [ 
 [0,1,0], 
 [0,1,0], 
 [0,1,0] 
 ], 
 [ 
 [0,0,0], 
 [1,1,1], 
 [0,0,0] 
 ] 
 ] 
 ]; 
 
 #当前的方块 
 curBrick = None; 
 #当前方块数组 
 arr = None; 
 #当前方块形状 
 shape = -1; 
 #当前方块的行和列(最左上角) 
 curRow = -10; 
 curCol = -10; 
 
 #背景 
 back = list(); 
 #格子 
 gridBack = list(); 
 
 #初始化 
 def init(self): 
 
 for i in range(0,self.rows): 
 
 self.back.insert(i,list()); 
 self.gridBack.insert(i,list()); 
 
 for i in range(0,self.rows): 
 
 for j in range(0,self.cols): 
 
 self.back[i].insert(j,0); 
 self.gridBack[i].insert(j,self.canvas.create_rectangle(30*j,30*i,30*(j+1),30*(i+1),fill="black")); 
 
 #绘制游戏的格子 
 def drawRect(self): 
 
 for i in range(0,self.rows): 
 
 for j in range(0,self.cols): 
 
 
 if self.back[i][j]==1: 
 
 self.canvas.itemconfig(self.gridBack[i][j],fill="blue",outline="white"); 
 
 elif self.back[i][j]==0: 
 
 self.canvas.itemconfig(self.gridBack[i][j],fill="black",outline="white"); 
 
 
 #绘制当前正在运动的方块 
 if self.curRow!=-10 and self.curCol!=-10: 
 
 for i in range(0,len(self.arr)): 
 
 for j in range(0,len(self.arr[i])): 
 
 if self.arr[i][j]==1: 
 
 self.canvas.itemconfig(self.gridBack[self.curRow+i][self.curCol+j],fill="blue",outline="white"); 
 
 #判断方块是否已经运动到达底部 
 if self.isDown: 
 
 for i in range(0,3): 
 
 for j in range(0,3): 
 
 if self.arr[i][j]!=0: 
 
 self.back[self.curRow+i][self.curCol+j] = self.arr[i][j]; 
 
 #判断整行消除 
 self.removeRow(); 
 
 #获得下一个方块 
 self.getCurBrick(); 
 
 #判断是否有整行需要消除 
 def removeRow(self): 
 
 for i in range(0,self.rows): 
 
 tag1 = True; 
 for j in range(0,self.cols): 
 
 if self.back[i][j]==0: 
 
 tag1 = False; 
 break; 
 
 if tag1==True: 
 
 #从上向下挪动 
 for m in xrange(i-1,0,-1): 
 
 for n in range(0,self.cols): 
 
 self.back[m+1][n] = self.back[m][n]; 
 
 #获得当前的方块 
 def getCurBrick(self): 
 
 self.curBrick = randint(0,len(self.brick)-1); 
 self.shape = 0; 
 #当前方块数组 
 self.arr = self.brick[self.curBrick][self.shape]; 
 
 self.curRow = 0; 
 self.curCol = 1; 
 
 #是否到底部为False 
 self.isDown = False; 
 
 #监听键盘输入 
 def onKeyboardEvent(self,event): 
 
 #未开始,不必监听键盘输入 
 if self.start == False: 
 
 return; 
 
 #记录原来的值 
 tempCurCol = self.curCol; 
 tempCurRow = self.curRow; 
 tempShape = self.shape; 
 tempArr = self.arr; 
 direction = -1; 
 
 if event.keycode==37: 
 
 #左移 
 self.curCol-=1; 
 direction = 1; 
 elif event.keycode==38: 
 #变化方块的形状 
 self.shape+=1; 
 direction = 2; 
 
 if self.shape>=4: 
 
 self.shape=0; 
 self.arr = self.brick[self.curBrick][self.shape]; 
 elif event.keycode==39: 
 
 direction = 3; 
 #右移 
 self.curCol+=1; 
 elif event.keycode==40: 
 
 direction = 4; 
 #下移 
 self.curRow+=1; 
 
 if self.isEdge(direction)==False: 
 
 self.curCol = tempCurCol; 
 self.curRow = tempCurRow; 
 self.shape = tempShape; 
 self.arr = tempArr; 
 
 self.drawRect(); 
 
 return True; 
 
 #判断当前方块是否到达边界 
 def isEdge(self,direction): 
 
 tag = True; 
 
 #向左,判断边界 
 if direction==1: 
 
 for i in range(0,3): 
 
 for j in range(0,3): 
 
 if self.arr[j][i]!=0 and (self.curCol+i<0 or self.back[self.curRow+j][self.curCol+i]!=0): 
 
 tag = False; 
 break; 
 #向右,判断边界 
 elif direction==3: 
 
 for i in range(0,3): 
 
 for j in range(0,3): 
 
 if self.arr[j][i]!=0 and (self.curCol+i>=self.cols or self.back[self.curRow+j][self.curCol+i]!=0): 
 
 tag = False; 
 break; 
 #向下,判断底部 
 elif direction==4: 
 
 for i in range(0,3): 
 
 for j in range(0,3): 
 
 if self.arr[i][j]!=0 and (self.curRow+i>=self.rows or self.back[self.curRow+i][self.curCol+j]!=0): 
 
 tag = False; 
 self.isDown = True; 
 break; 
 #进行变形,判断边界 
 elif direction==2: 
 
 if self.curCol<0: 
 
 self.curCol=0; 
 
 if self.curCol+2>=self.cols: 
 
 self.curCol = self.cols-3; 
 
 if self.curRow+2>=self.rows: 
 
 self.curRow = self.curRow-3; 
 
 
 return tag; 
 
 #方块向下移动 
 def brickDown(self): 
 
 while True: 
 
 if self.start==False: 
 
 print("exit thread"); 
 break; 
 
 tempRow = self.curRow; 
 self.curRow+=1; 
 
 if self.isEdge(4)==False: 
 
 self.curRow = tempRow; 
 
 self.drawRect(); 
 
 #每一秒下降一格 
 sleep(1); 
 
 #运行 
 def __init__(self): 
 
 self.window = Tk(); 
 self.window.title(self.title); 
 self.window.minsize(self.width,self.height); 
 self.window.maxsize(self.width,self.height); 
 
 self.frame1 = Frame(self.window,width=300,height=600,bg="black"); 
 self.frame1.place(x=20,y=30); 
 
 self.canvas = Canvas(self.frame1,width=300,height=600,bg="black"); 
 
 self.init(); 
 
 #获得当前的方块 
 self.getCurBrick(); 
 
 #按照数组,绘制格子 
 self.drawRect(); 
 
 self.canvas.pack(); 
 
 #监听键盘事件 
 self.window.bind("<KeyPress>",self.onKeyboardEvent); 
 
 #启动方块下落线程 
 downThread = threading.Thread(target=self.brickDown,args=()); 
 downThread.start(); 
 
 self.window.mainloop(); 
 
 self.start=False; 
 
 pass; 
 
if __name__=='__main__': 
 
 brickGame = BrickGame();

估计用图形界面会很少,因为本人是WEB开发。不过,怎样也抑制不住这颗喜欢写游戏的心啊!

更多俄罗斯方块精彩文章请点击专题:俄罗斯方块游戏集合 进行学习。

更多关于python游戏的精彩文章请点击查看以下专题:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python利用elaphe制作二维条形码实现代码
May 25 Python
探索Python3.4中新引入的asyncio模块
Apr 08 Python
Django框架下在URLconf中指定视图缓存的方法
Jul 23 Python
详解Python函数可变参数定义及其参数传递方式
Aug 02 Python
Python面向对象之类的定义与继承用法示例
Jan 14 Python
Python计算时间间隔(精确到微妙)的代码实例
Feb 26 Python
详解Python中的测试工具
Jun 09 Python
python线程的几种创建方式详解
Aug 29 Python
python GUI库图形界面开发之PyQt5多行文本框控件QTextEdit详细使用方法实例
Feb 28 Python
Python实现爬取并分析电商评论
Jun 19 Python
PyQt中使用QtSql连接MySql数据库的方法
Jul 28 Python
如何利用python实现列表嵌套字典取值
Jun 10 Python
python中的变量如何开辟内存
Jun 26 #Python
pyQt4实现俄罗斯方块游戏
Jun 26 #Python
朴素贝叶斯分类算法原理与Python实现与使用方法案例
Jun 26 #Python
python实现俄罗斯方块
Jun 26 #Python
解决python报错MemoryError的问题
Jun 26 #Python
pygame实现俄罗斯方块游戏
Jun 26 #Python
python和pygame实现简单俄罗斯方块游戏
Feb 19 #Python
You might like
PHP安全配置详细说明
2011/09/26 PHP
解析PHP 使用curl提交json格式数据
2013/06/29 PHP
PHP中常用的转义函数
2014/02/28 PHP
PHP 断点续传实例详解
2017/11/11 PHP
浅析PHP开发规范
2018/02/05 PHP
JavaScript DOM学习第八章 表单错误提示
2010/02/19 Javascript
jQuery创建插件的代码分析
2011/04/14 Javascript
SharePoint 客户端对象模型 (一) ECMA Script
2011/05/22 Javascript
使用CSS样式position:fixed水平滚动的方法
2014/02/19 Javascript
NodeJS学习笔记之Connect中间件应用实例
2015/01/27 NodeJs
实例讲解使用原生JavaScript处理AJAX请求的方法
2016/05/10 Javascript
手机端图片缩放旋转全屏查看PhotoSwipe.js插件实现
2016/08/25 Javascript
Bootstrap modal使用及点击外部不消失的解决方法
2016/12/13 Javascript
使用原生的javascript来实现轮播图
2017/02/24 Javascript
Vue.js中的computed工作原理
2018/03/22 Javascript
详解Vue2.0配置mint-ui踩过的那些坑
2018/04/23 Javascript
基于js Canvas实现二次贝塞尔曲线
2018/12/25 Javascript
简单了解Javscript中兄弟ifream的方法调用
2019/06/17 Javascript
JS实现纵向轮播图(初级版)
2020/01/18 Javascript
vue.js this.$router.push获取不到params参数问题
2020/03/03 Javascript
Node 使用express-http-proxy 做api网关的实现
2020/10/15 Javascript
使用python实现扫描端口示例
2014/03/29 Python
全面解读Python Web开发框架Django
2014/06/30 Python
python实现弹窗祝福效果
2019/04/07 Python
Python 处理文件的几种方式
2019/08/23 Python
如何搭建pytorch环境的方法步骤
2020/05/06 Python
python_matplotlib改变横坐标和纵坐标上的刻度(ticks)方式
2020/05/16 Python
python 6行代码制作月历生成器
2020/09/18 Python
Omio法国:全欧洲低价大巴、火车和航班搜索和比价
2017/11/13 全球购物
《神奇的克隆》教学反思
2014/04/10 职场文书
日语系毕业求职信
2014/07/27 职场文书
股东出资证明书范例
2014/10/04 职场文书
债务纠纷起诉书
2015/05/20 职场文书
2015年党小组工作总结
2015/05/26 职场文书
2015年中秋晚会主持稿
2015/07/30 职场文书
使用 DataAnt 监控 Apache APISIX的原理解析
2022/07/07 Servers