承接上一部,上一部講到實現了添加游戲框,游戲背景,以及玩家飛機的運動

這一次的教程的目的在於實現生成敵機、敵機的移動、敵機與玩家的碰撞后,兩者的摧毀,重生等
如下:

先准備資源,網絡上也可以下載到,我這里直接用的教程的資源,為什么呢(確實好看,自己找的慘不忍睹)





然后不多說,直接上代碼,基本上每一行能進行的注釋以及方法結束都寫得很詳細了,如果有描述的不是很正確的地方,可以評論一下(我是小菜雞,請大佬輕點噴)
enemy.py(敵機類,包含敵機的屬性、運行、重置等)
其中的知識點是1.”基類的初始化“
import pygame from random import * # 小型飛機類 # pygame.sprite.Sprite就是Pygame里面用來實現精靈的一個類,使用時,並不需要對它實例化,只需要繼承他,然后按需寫出自己的類就好了,因此非常簡單實用 # 所有精靈在建立時都是從pygame.sprite.Sprite中繼承的 class SmallEnemy(pygame.sprite.Sprite): def __init__(self, bg_size): # 這里涉及到”基類的初始化“建用百度好好理解一下。 # 通俗來說,SmallEnemy類繼承了父類pygame.sprite.Sprite,子類父類都有__init__()這個函數,如果子類不實現這個init()函數, # 那么初始化時直接調用父類的初始化函數,如果子類實現這個init()函數,就覆蓋了父類的這個函數,既然繼承父類,就要在這個函數里顯式調用一下父類的__init__() pygame.sprite.Sprite.__init__(self) # 敵機圖片 self.image = pygame.image.load('images/enemy1.png').convert_alpha() # 敵機摧毀圖片 self.destory_images = [] self.destory_images.extend([ pygame.image.load('images/enemy1_down1.png').convert_alpha(), pygame.image.load('images/enemy1_down2.png').convert_alpha(), pygame.image.load('images/enemy1_down3.png').convert_alpha(), pygame.image.load('images/enemy1_down4.png').convert_alpha() ]) # 定義屏幕寬高 self.width = bg_size[0] self.height = bg_size[1] # get_rect()是一個處理矩形圖像的方法,返回值包含矩形的各屬性,這里返回敵機圖片的位置,可以獲取圖片的寬高等屬性 self.rect = self.image.get_rect() # 隨機生成飛機的位置,randint(a,b)即生成a<=n<=b,即在屏幕寬度,以及負5倍的高度下隨機生成 self.rect.left = randint(0, self.width - self.rect.width) self.rect.top = randint(-5 * self.height, 0) # 小型敵機速度 self.speed = 2 # 敵機是否存活狀態 self.active = True # 飛機碰撞檢測,會忽略掉圖片中白色的背景部分 self.mask = pygame.mask.from_surface(self.image) # 小型敵機向下移動 def samll_enemy_move(self): # 飛機還未飛出屏幕外,就向下運動(這里可以自己修改飛機的飛行軌跡,比如x軸的隨機左右飛行等,怎么浪怎么來) if self.rect.top < self.height: self.rect.top += self.speed else: self.reset() # 飛機飛出屏幕外后,別浪費,重置其位置(或者摧毀也行,反正得處理,不然內存會炸) def reset(self): # 這里寫不寫這個感覺都行,因為都是存活狀態,寫了更保險 self.active = True # 隨機重置飛機的位置,跟上面生成得一樣 self.rect.left = randint(0, self.width - self.rect.width) self.rect.top = randint(-5 * self.height, 0)
這里的mian是在上一部的基礎上添加的,即調用敵機類,生成小型敵機,然后敵機運動,如果與玩家飛機碰撞,則兩者都摧毀,且玩家生命減一
main.py
主要涉及的知識點:
1.pygame.sprite.Group()函數可以創建一個精靈組,從而統一管理,以及Group對應的方法
2.pygame.USEREVENT代表事件1,pygame.time.set_timer:就是每隔一段時間(這里是3毫秒 * 1000 = 3s),去執行一些動作,然后通過event.type == invincible_event去捕獲事件的發生
3.碰撞檢測,pygame.sprite.spritecollide(sprite,sprite_group,bool):一個組中的所有精靈都會逐個地對另外一個單個精靈進行沖突檢測
import pygame import sys import traceback from pygame.locals import * from random import * import myplane import enemy # 初始化 pygame.init() # 設置窗口大小 bg_size = width, height = 400, 700 # 實際上是元組 screen = pygame.display.set_mode(bg_size) # 設置窗口 pygame.display.set_caption("飛機大戰") # 窗口標題 # 加載背景圖片,對於普通圖像的顯示效果有沒有convert都是一樣的,但是 使用 convert 可以轉換格式,提高 blit 的速度 background = pygame.image.load("images/background.png").convert() # 設置黑、綠、紅、百幾種顏色對應值,后面會用到 BLACK = (0, 0, 0) GREEN = (0, 255, 0) RED = (255, 0, 0) WHITE = (255, 255, 255) # 生成敵方小型飛機 def add_small_enemy(small_enemies, enemiesGroup, num): for i in range(num): smallenemy = enemy.SmallEnemy(bg_size) # 精靈組來實現多個圖像,很適合處理精靈列表,有添加,移除,繪制,更新等方法 # Group.sprites 精靈組 # Group.copy 復制 # Group.add 添加 # Group.remove 移除 # Group.has 判斷精靈組成員 # Group.update 更新 # Group.draw 位塊顯示 # Group.clear - 繪制背景 # Group.empty 清空 # 將這一組敵機都添加上小型飛機屬性,相當於統一處理,統一賦值 small_enemies.add(smallenemy) enemiesGroup.add(smallenemy) def main(): # 創建時鍾對象(可以控制游戲循環頻率) clock = pygame.time.Clock() # 生成玩家飛機 me = myplane.MyPlane(bg_size) # 存放所有敵方飛機,這個飛機組包含了小型飛機、中型飛機、大型飛機的各種屬性,只要用於處理碰撞 # 當程序中有大量的實體的時候,操作這些實體將會是一件相當麻煩的事 # 使用pygame.sprite.Group()函數可以創建一個精靈組,從而統一管理,這里創建了一個敵機組 enemiesGroup = pygame.sprite.Group() # 生成地方小型飛機,敵方小型飛機也是一個組,進行統一處理 small_enemies = pygame.sprite.Group() add_small_enemy(small_enemies, enemiesGroup, 15) # 玩家三條命 life_num = 3 # 設置無敵時間事件,pygame.USEREVENT代表事件1,pygame.USEREVENT+1代表事件2,以此類推,這里相當於定義了一個事件 invincible_event = pygame.USEREVENT # 游戲暫停,默認為非暫停狀態 paused = False # 控制玩家飛機圖片切換,展示突突突的效果 switch_image = True # 切換延時 delay = 100 # 游戲分數 score = 0 # 飛機爆炸的圖片下標,依次為小型敵機,中型敵機,大型敵機,玩家飛機的爆炸的圖片的下標,切換下標來改變爆炸圖片 e1_destory_index = 0 e2_destory_index = 0 e3_destory_index = 0 me_destory_index = 0 running = True while running: # 獲取事件 for event in pygame.event.get(): # 結束事件觸發結束操作 if event.type == QUIT: pygame.quit() sys.exit() # 在觸發碰撞的時候,寫了pygame.time.set_timer(invincible_event, 3*1000) # 意思就是3秒后將會執行invincible_event事件,這里捕獲了invincible_event事件,執行后,將取消這個計時器,防止循環重復執行,等待下一次觸發 if event.type == invincible_event: # 解除無敵狀態 me.invincible = False pygame.time.set_timer(invincible_event, 0) # 檢測用戶鍵盤操作,分別為上下左右 key_pressed = pygame.key.get_pressed() if key_pressed[K_w] or key_pressed[K_UP]: me.moveUp() if key_pressed[K_s] or key_pressed[K_DOWN]: me.moveDown() if key_pressed[K_a] or key_pressed[K_LEFT]: me.moveLeft() if key_pressed[K_d] or key_pressed[K_RIGHT]: me.moveRight() # 在屏幕上面繪制背景圖像,並指定位置 screen.blit(background, (0, 0)) # 繪制子彈補給、炸彈補給等各種元素 # 未暫停且生命大於0 if paused == False and life_num > 0: # 繪制小型敵機,這里是由於上面定義了小型飛機組,飛機組add了小型飛機屬性(速度、位置等),這時候地圖上就生成了飛機 # 如果這些飛機屬於小型敵機類,即一起處理 for ei in small_enemies: # 敵機是活的,未被擊毀 if ei.active == True: # 繪制小型敵機,並且敵機開始運動 screen.blit(ei.image, ei.rect) ei.samll_enemy_move() # 小型敵機被摧毀(被玩家擊毀或者與玩家碰撞) else: # 這里設置delay % 4是指爆炸畫面為4幀(個人猜測),理解為爆炸停留時間,可自行設置 if not (delay % 4): # 用於播放爆炸聲音,每一架敵機只有一次 if e1_destory_index == 0: print("播放敵機爆炸聲音") # 繪制敵機撞擊爆炸畫面 screen.blit(ei.destory_images[e1_destory_index], ei.rect) # 切換爆炸圖片下標,從而切換爆炸圖片 e1_destory_index = (e1_destory_index + 1) % 4 # 經歷完一輪爆炸的敵機,可以將其銷毀,也可以重生,都是不能不處理,不然會一直爆炸、爆炸 # 這里選擇將其重生 if e1_destory_index == 0: ei.reset() score += 1000 print("得分:", score) # 做碰撞檢測,pygame.sprite.spritecollide(sprite,sprite_group,bool):一個組中的所有精靈都會逐個地對另外一個單個精靈進行沖突檢測,發生沖突的精靈會作為一個列表返回。 # 第一個參數就是單個精靈,第二個參數是精靈組,第三個參數是一個bool值,最后這個參數起了很大的作用。當為True的時候,會刪除組中所有沖突的精靈,False的時候不會刪除沖突的精靈 # 第四個參數是:兩個精靈之間的像素遮罩檢測 enemy_collide = pygame.sprite.spritecollide(me, enemiesGroup, False, pygame.sprite.collide_mask) # 碰撞處理,如果不是無敵狀態下發生碰撞 if enemy_collide and not me.invincible: # 玩家飛機觸發摧毀狀態 me.active = False # enemy_collide是一個列表,存儲所有跟玩家飛機發生碰撞的敵機,然后把碰撞的敵機狀態置為摧毀狀態 for ei in enemy_collide: ei.active = False # 繪制玩家飛機,如果飛機為激活狀態 if me.active: # 在屏幕上繪制玩家飛機,switch_image為是否切換圖片 if switch_image: screen.blit(me.image1, me.rect) # 切換一下飛行圖片 else: screen.blit(me.image2, me.rect) # 代表飛機遭到碰撞,激活爆炸事件 else: if not (delay % 4): # 用於播放爆炸聲音,每一架敵機只有一次 if me_destory_index == 0: print("玩家飛機爆炸聲音") # 繪制玩家撞擊爆炸畫面 screen.blit(me.destory_image[me_destory_index], me.rect) # 切換爆炸圖片下標,從而切換爆炸圖片 me_destory_index = (me_destory_index + 1) % 4 # 爆炸畫面播放完之后飛機重生 if me_destory_index == 0: # 生命減一條,如果見到0,會自動跳過上一級循環 life_num -= 1 # 重置狀態 me.reset() # 無敵時間設置為3秒,3秒后,觸發無敵時間事件,pygame.time.set_timer:就是每隔一段時間(這里是3毫秒 * 1000 = 3s),去執行一些動作 pygame.time.set_timer(invincible_event, 3 * 1000) delay -= 1 if delay == 0: delay = 100 # 每5幀切換一下飛行圖片樣式 if delay % 5 == 0: switch_image = not switch_image # 更新整個待顯示的 Surface 對象到屏幕上,將內存中的內容顯示到屏幕上 pygame.display.flip() # 通過時鍾對象指定循環頻率,每秒循環60次 # 幀速率是指程序每秒在屏幕山繪制圖 clock.tick(60) if __name__ == "__main__": try: main() # 服務正常退出 except SystemExit: print("游戲正常退出!") # pass忽略錯誤並繼續往下運行,其實這里以及退出了 pass # 服務出現其他的異常 except: # 直接將錯誤打印出來 traceback.print_exc() pygame.quit()
上一部的myplane.py,這里沒有添加其他額外的東西了,這里也貼上去:
import pygame # 玩家飛機類,pygame.sprite模塊里面包含了一個名為Sprite類,他是pygame本身自帶的一個精靈。 class MyPlane(pygame.sprite.Sprite): def __init__(self, bg_size): # convert_alpha()更改圖像的像素格式,包括每個像素的alpha,相當於圖片背景變為透明 self.image1 = pygame.image.load('images/me1.png').convert_alpha() self.image2 = pygame.image.load('images/me2.png').convert_alpha() # 飛機摧毀圖片,以數字形式保存 self.destory_image = [] self.destory_image.extend([ pygame.image.load('images/me_destroy_1.png').convert_alpha(), pygame.image.load('images/me_destroy_2.png').convert_alpha(), pygame.image.load('images/me_destroy_3.png').convert_alpha(), pygame.image.load('images/me_destroy_4.png').convert_alpha() ]) # 定義屏幕寬高 self.width = bg_size[0] self.height = bg_size[1] # get_rect()是一個處理矩形圖像的方法,返回值包含矩形的各屬性,這里返回飛機圖片1的位置,可以獲取圖片的寬高等屬性 self.rect = self.image1.get_rect() # 飛機的初始化位置,//是整除,位置居中以及高度為圖片下框離屏幕最下方60 self.rect.left = (self.width - self.rect.width)//2 self.rect.top = self.height - self.rect.height - 60 # 設置飛機的速度 self.myPlaneSpeed = 10 self.active = True # 設置飛機是否是無敵狀態(重生3秒內無敵) self.invincible = False # 飛機碰撞檢測,會忽略掉圖片中白色的背景部分,從指定 Surface 對象中返回一個 Mask # 用於快速實現完美的碰撞檢測,Mask 可以精確到 1 個像素級別的判斷。 # Surface 對象中透明的部分設置為 1,不透明部分設置為 0。 self.mask = pygame.mask.from_surface(self.image1) # 玩家飛機向上移動 def moveUp(self): # 說明還沒定格,即還未到達游戲界面上邊界 if self.rect.top > 0: self.rect.top -= self.myPlaneSpeed # 說明移動到達上邊界了 else: self.rect.top = 0 # 玩家飛機向下移動 def moveDown(self): # 底部需要划出60的高度用來展示其他數據(炸彈數,生命數等) if self.rect.bottom < self.height - 60: # self.rect.bottom指的是飛機圖片下邊界 self.rect.bottom += self.myPlaneSpeed else: self.rect.bottom = self.height - 60 # 玩家飛機向左移動 def moveLeft(self): if self.rect.left > 0: self.rect.left -= self.myPlaneSpeed else: self.rect.left = 0 # 玩家飛機向右移動 def moveRight(self): if self.rect.right < self.width: self.rect.right += self.myPlaneSpeed else: self.rect.right = self.width # 玩家飛機重生 def reset(self): self.active = True # 重生時處於無敵狀態 self.invincible = True # 重生飛機的初始化位置,//是整除,位置居中以及高度為圖片下框離屏幕最下方60 self.rect.left = (self.width - self.rect.width) // 2 self.rect.top = self.height - self.rect.height - 60
