Python:游戲:300行代碼實現俄羅斯方塊

来源:https://www.cnblogs.com/gl1573/archive/2019/01/04/10213812.html
-Advertisement-
Play Games

本文代碼基於 python3.6 和 pygame1.9.4。 俄羅斯方塊是兒時最經典的游戲之一,剛開始接觸 pygame 的時候就想寫一個俄羅斯方塊。但是想到旋轉,停靠,消除等操作,感覺好像很難啊,等真正寫完了發現,一共也就 300 行代碼,並沒有什麼難的。 先來看一個游戲截圖,有點醜,好吧,我沒 ...


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

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

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


現在來看一下實現的過程。

外形

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

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

import sys
import pygame
from pygame.locals import *

SIZE = 30  # 每個小方格大小
BLOCK_HEIGHT = 20  # 游戲區高度
BLOCK_WIDTH = 10   # 游戲區寬度
BORDER_WIDTH = 4   # 游戲區邊框寬度
BORDER_COLOR = (4040200)  # 游戲區邊框顏色
SCREEN_WIDTH = SIZE * (BLOCK_WIDTH + 5)  # 游戲屏幕的寬
SCREEN_HEIGHT = SIZE * BLOCK_HEIGHT      # 游戲屏幕的高
BG_COLOR = (404060)  # 背景色
BLACK = (000)


def print_text(screen, font, x, y, text, fcolor=(255255255)):
    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 // 20),
                         (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, 10f'得分: ')
        print_text(screen, font1, font_pos_x, 10 + font1_height + 6f'{score}')
        print_text(screen, font1, font_pos_x, 20 + (font1_height + 6) * 2f'速度: ')
        print_text(screen, font1, font_pos_x, 20 + (font1_height + 6) * 3f'{score // 10000}')
        print_text(screen, font1, font_pos_x, 30 + (font1_height + 6) * 4f'下一個:')

        pygame.display.flip()


if __name__ == '__main__':
    main()

方塊

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


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

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

例如豎條行,定義是:

['.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(00), Point(21), 'S'1),
           Block(['0..',
                  '00.',
                  '.0.'], Point(00), Point(12), '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):
        
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 上述按鈕CSS規則 上述按鈕CSS顏色對應樣式依次為 未完待續...... ...
  • 本文主要從8個章節詳解vue技術揭秘,小編覺得挺有用的,分享給大家。 為了把 Vue.js 的源碼講明白,課程設計成由淺入深,分為核心、編譯、擴展、生態四個方面去講,並拆成了八個章節,如下: ...
  • 你們有沒有遇到過這樣的情況,ES6看過了,Promise的文字概念都懂,但是我要怎麼在項目中去寫一個Promise呢? 那天我就是帶著這樣的疑問去網上搜了下。最後成功地在項目中應用了Promise,只有實際成功使用一次,才能明白它的前因後果,明白它的用途。 1.這是一個vue的電商項目-商品詳情頁 ...
  • 這個案例網上是沒有的,屬於無忌獨創,當時在幫孩子輔導作業,小學科學,裡面有一點內容是關於人的牙齒,說牙齒分為:門齒、犬齒、臼齒,問閨女,為什麼這麼分呢?閨女說牙齒雖然都是用來咬食物,但是食物種類很多,咬碎需要的工具也不一樣,門齒用來切割食物,如:蘋果、梨;犬齒用來撕碎食物,如肉類;臼齒用來磨碎食物,... ...
  • 公眾號:SAP Technical 本文作者:matinal 原文出處:http://www.cnblogs.com/SAPmatinal/ 原文鏈接:【MM系列】SAP MM 非限制/可用庫存 前言部分 今天簡單討論一下MM的非限制使用庫存和可用庫存,如果文章有不正確的地方,請及時留言指出,我會第 ...
  • 今天我們來談談 Dubbo XML 配置相關內容。關於這部分內容我打算分為以下幾個部分進行介紹: Dubbo XML Spring 自定義 XML 標簽解析 Dubbo 自定義 XML 標簽解析 DubboBeanDefinitionParser.parse() End Dubbo XML 在本小節 ...
  • 這篇翻譯的不好 如果你看API文檔中的數組篇,你會發現類型一般寫成List.的寫法表示通用類型的數組(未明確指定數組中的數據類型)。通常情況泛型類型用E,T,S,K,V表示。 Why use generics? 為什麼用泛型 泛型是類型安全的(意思是你必須指定數據的類型),但是它的寫法比硬編碼指定類 ...
  • 開場白說點東西: { 抓住客戶的痛點、癢點、爽點,提出我們產品的核心價值。 產品定位 技術架構 以微服務為核心的前後端分離,業務積木裝配式技術架構。感測器採集,物聯網+互聯網轉換,大數據分散式、存儲、計算、可視化加持。消息引擎、搜索引擎、工作流引擎全方位技術支持。 研發模式 Scrum敏捷研發,讓每 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...