300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

嗨學程式設計發表於2019-01-04

本文程式碼基於 python3.6 和 pygame1.9.4。

俄羅斯方塊是兒時最經典的遊戲之一,剛開始接觸 pygame 的時候就想寫一個俄羅斯方塊。但是想到旋轉,停靠,消除等操作,感覺好像很難啊,等真正寫完了發現,一共也就 300 行程式碼,並沒有什麼難的。

先來看一個遊戲截圖,有點醜,好吧,我沒啥美術細胞,但是主體功能都實現了,可以玩起來。

本人對於Python學習建立了一個小小的學習圈子,為各位提供了一個平臺,大家一起來討論學習Python。歡迎各位到來Python學習群:960410445一起討論視訊分享學習。Python是未來的發展方向,正在挑戰我們的分析能力及對世界的認知方式,因此,我們與時俱進,迎接變化,並不斷的成長,掌握Python核心技術,才是掌握真正的價值所在。
300行Python程式碼實現俄羅斯方塊,致敬逝去的童年


外形

俄羅斯方塊整個介面分為兩部分,一部分是左邊的遊戲區域,另一部分是右邊的顯示區域,顯示得分、速度、下一個方塊樣式等。這裡就不放截圖了,看上圖就可以。

遊戲區域跟貪吃蛇一樣,是由一個個小方格組成的,為了看得直觀,我特意畫了網格線。

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年


方塊

接下來就是要定義方塊,方塊的形狀一共有以下 7 種:

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

I 型

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

O 型

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

T 型

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

S 型

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

Z 型

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

L 型

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

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 型的豎條,在緊貼左右邊框的時候是不可以旋轉的。這點我有印象,可以肯定。但是對於其他的形狀,我就不是很確定了,我百度搜了下,找了個網頁版的俄羅斯方塊玩了下,發現也是不可以的。例如:

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年

在緊貼右邊框的時候是無法旋轉的。如果要每一個形狀都去判斷一下,那實在是太煩了。從方塊的定義入手,就可以很簡單的實現。

例如豎條行,定義是:

['.0..',

'.0..',

'.0..',

'.0..']

豎條是可以貼邊的,所以當它在最左邊的時候,X 軸座標是 -1,這是因為定義中左邊一豎排是空的。我們只需判定,當方塊所定義的形狀(包括空的部分)完全在遊戲區域內時才可以旋轉。

我之前所說,全都定義成 4 × 4 不好,原因就在這裡,對於 T 型等其他形狀,無法做這個判定。所以,對於 T 型等形狀,我們可以定義成 3 × 3 的格式:

['.0.',

'000',

'...']

還有一種情況是無法旋轉的,就是旋轉後的位置已經被別的方塊佔了。另外下落,左右移動,都要做這個判斷。既然這些是一致的,那麼就可以用同一個方法來判斷。

先要定義一個 game_area 變數,用於存放整個遊戲區域當前的狀態:

game_area = [['.'] * BLOCK_WIDTHfor_inrange(BLOCK_HEIGHT)]

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

另外,需要一些變數定義當前下落方塊的狀態

cur_block= None# 當前下落方塊

cur_pos_x, cur_pos_y = 0, 0 # 當前下落方塊的座標

方塊我們是以二維陣列的方式定義的,並且存在空行和空列,如果我們遍歷這個二維陣列判斷其所在的區域在當前遊戲區域內是否已經被別的方塊所佔,這個是可以實現的。我們考慮另外一種情況,一個豎條形,左邊一排是空的,這空的一排是可以移出遊戲區域的,這個怎麼判斷?每次左移的時候都去判斷一下左邊一排全都是空嗎?這太麻煩了。並且方塊都是固定的,所以這些我們可以提前定義好。最終方塊定義如下:

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年


方塊需要包含兩個方法,獲取隨機一個方塊和旋轉時獲取旋轉後的方塊

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年


判斷是否可以旋轉,下落,移動的方法也很容易實現了

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年


停靠

最後一個問題是停靠,當方塊下落到底或者遇到別的方塊之後,就不能在下落了。我將此稱之為“停靠”,有個名字說起來也方便一點。

首先是要判斷是否可以停靠,停靠發生之後,就是將當前方塊的非空點畫到遊戲區域上,說白了,就是將 cur_block 的非空點按對應位置複製到 game_area 裡去。並且計算是否有一排被全部填滿了,全部填滿則消除。

300行Python程式碼實現俄羅斯方塊,致敬逝去的童年


至此,整個俄羅斯方塊的主體功能就算是完成了。

這裡很多引數是可以調的,例如覺得旋轉彆扭,可以直接調整方塊的定義,而無需去改動程式碼邏輯。


相關文章