python實(shí)現(xiàn)俄羅斯方塊游戲
在公司實(shí)習(xí)。公司推崇Python和Django框架,所以也得跟著學(xué)點(diǎn)。
簡單瞅了下Tkinter,和Canvas配合在一起,還算是簡潔的界面開發(fā)API。threading.Thread創(chuàng)建新的線程,其多線程機(jī)制也算是方便。
只是canvas.create_rectangle居然不是繪制矩形,而是新建了矩形控件這點(diǎn)讓人大跌眼鏡。先開始,在線程里每次都重繪多個(gè)矩形(隨數(shù)組變化),其實(shí)是每次都新建了N個(gè)矩形,結(jié)果內(nèi)存暴增。原來,對矩形進(jìn)行變更時(shí),只需用canvas.itemconfig即可。
下面就是截圖(時(shí)間太晚,明日還得上班,做得非常粗糙...沒事時(shí)再慢慢修正)。
而代碼如下:
#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; #是否到達(dá)底部 isDown = True; #窗體 window = None; #frame frame1 = None; #繪圖類 canvas = None; #標(biāo)題 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] ] ] ]; #當(dāng)前的方塊 curBrick = None; #當(dāng)前方塊數(shù)組 arr = None; #當(dāng)前方塊形狀 shape = -1; #當(dāng)前方塊的行和列(最左上角) 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"); #繪制當(dāng)前正在運(yùn)動的方塊 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"); #判斷方塊是否已經(jīng)運(yùn)動到達(dá)底部 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(); #獲得下一個(gè)方塊 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]; #獲得當(dāng)前的方塊 def getCurBrick(self): self.curBrick = randint(0,len(self.brick)-1); self.shape = 0; #當(dāng)前方塊數(shù)組 self.arr = self.brick[self.curBrick][self.shape]; self.curRow = 0; self.curCol = 1; #是否到底部為False self.isDown = False; #監(jiān)聽鍵盤輸入 def onKeyboardEvent(self,event): #未開始,不必監(jiān)聽鍵盤輸入 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; #判斷當(dāng)前方塊是否到達(dá)邊界 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; #進(jìn)行變形,判斷邊界 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); #運(yùn)行 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(); #獲得當(dāng)前的方塊 self.getCurBrick(); #按照數(shù)組,繪制格子 self.drawRect(); self.canvas.pack(); #監(jiān)聽鍵盤事件 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();
估計(jì)用圖形界面會很少,因?yàn)楸救耸荳EB開發(fā)。不過,怎樣也抑制不住這顆喜歡寫游戲的心啊!
更多俄羅斯方塊精彩文章請點(diǎn)擊專題:俄羅斯方塊游戲集合 進(jìn)行學(xué)習(xí)。
更多關(guān)于python游戲的精彩文章請點(diǎn)擊查看以下專題:
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
PyCharm安裝庫numpy失敗問題的詳細(xì)解決方法
今天使用pycharm編譯python程序時(shí),由于要調(diào)用numpy包,但又未曾安裝numpy,于是就根據(jù)pycharm的提示進(jìn)行安裝,最后竟然提示出錯(cuò),下面這篇文章主要給大家介紹了關(guān)于PyCharm安裝庫numpy失敗問題的詳細(xì)解決方法,需要的朋友可以參考下2022-06-06使用實(shí)現(xiàn)XlsxWriter創(chuàng)建Excel文件并編輯
今天小編就為大家分享一篇使用實(shí)現(xiàn)XlsxWriter創(chuàng)建Excel文件并編輯,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-05-05Python?網(wǎng)頁請求之requests庫的使用詳解
requests?是?Python?中比較常用的網(wǎng)頁請求庫,主要用來發(fā)送?HTTP?請求,在使用爬蟲或測試服務(wù)器響應(yīng)數(shù)據(jù)時(shí)經(jīng)常會用到,使用起來十分簡潔,這篇文章主要介紹了Python?網(wǎng)頁請求之requests庫的使用詳解,需要的朋友可以參考下2022-09-09Python中numpy模塊常見用法demo實(shí)例小結(jié)
這篇文章主要介紹了Python中numpy模塊常見用法,結(jié)合實(shí)例形式總結(jié)分析了numpy常見的運(yùn)算操作技巧與注意事項(xiàng),需要的朋友可以參考下2019-03-03python通過pil模塊將raw圖片轉(zhuǎn)換成png圖片的方法
這篇文章主要介紹了python通過pil模塊將raw圖片轉(zhuǎn)換成png圖片的方法,實(shí)例分析了Python中pil模塊的使用技巧,并Image.fromstring函數(shù)進(jìn)行了較為詳盡的分析說明,需要的朋友可以參考下2015-03-03Python編程OpenCV和Numpy圖像處理庫實(shí)現(xiàn)圖片去水印
這篇文章主要介紹了Python編程中如何實(shí)現(xiàn)圖片去水印本文采用了OpenCV和Numpy的圖像處理的方法來實(shí)現(xiàn),文中附含詳細(xì)示例代碼,有需要的朋友可以借鑒參考下2021-09-09