日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > python >内容正文

python

python编写小游戏代码_Python小游戏之300行代码实现俄罗斯方块

發布時間:2024/3/24 python 27 豆豆
生活随笔 收集整理的這篇文章主要介紹了 python编写小游戏代码_Python小游戏之300行代码实现俄罗斯方块 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

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

來源:中文源碼網 瀏覽: 次 日期:2019年11月5日

【下載文檔: Python小游戲之300行代碼實現俄羅斯方塊.txt 】

(友情提示:右鍵點上行txt文檔名->目標另存為)

Python小游戲之300行代碼實現俄羅斯方塊前言

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

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

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

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

外形

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

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

import sys

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

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 namedtuplePoint = 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停靠

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

首先是要判斷是否可以???#xff0c;??堪l生之后,就是將當前方塊的非空點畫到游戲區域上,說白了,就是將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至此,整個俄羅斯方塊的主體功能就算是完成了。

這里很多參數是可以調的,例如覺得旋轉別扭,可以直接調整方塊的定義,而無需去改動代碼邏輯。

源碼下載:http://xiazai.zwyuanma.com/201901/yuanma/python-Tetris.rar

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對中文源碼網的支持。

親,試試微信掃碼分享本頁! *^_^*

總結

以上是生活随笔為你收集整理的python编写小游戏代码_Python小游戏之300行代码实现俄罗斯方块的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。