亲宝软件园·资讯

展开

python之pygame模块实现飞机大战完整代码

繁华落尽,寻一世真情 人气:0

Python飞机大战步骤:

1.数据区
2.主界面
3.飞船
4.事件监控及边界
5.外星人
6.记分系统

飞机大战效果图:

源码:

"""
功能:飞机大战
time:2019/10/3

"""
import os
import pygame
import sys
import time
from pygame.sprite import Sprite, Group

"""
1.定义主界面
2.定义飞船位置
3.边界及键盘操作
4.记分系统
"""
#1.数据区
#定义一个参数类
class Settings():
 def __init__(self):
 #屏幕设置
 self.screen_width = 1100
 self.screen_height = 600
 self.background = (230,230,230)
 self.background_image = pygame.image.load("C:/Users/Administrator/Desktop/xxx.jpg")
 #子弹设置
 self.bullet_width = 3
 self.bullet_height = 15
 self.bullet_color = 60,60,60
 # 屏幕上子弹的个数
 self.bullets_allow = 3
 #外星人设置
 self.fleet_drop_speed = 10
 self.ship_limit = 3
 #玩家升级后加快游戏速度
 self.speed_scale = 1.2
 #外星人点数提高的速度
 self.score_speed = 1.5
 #初始化随游戏变化的属性
 self.init_setting()

 #每个外星人的分数
 self.alien_points = 50
 def init_setting(self): #初始化随游戏变化的属性
 self.ship_speed = 1.6
 self.bullet_speed_factor = 2.5
 self.alien_speed_factor = 1

 #设置左右移动的标志,1右移,-1左移
 self.fleet_direction =1
 def increase_speed(self):
 #提高速度设置和外星人点数
 self.ship_speed *= self.speed_scale
 self.bullet_speed_factor *= self.speed_scale
 self.alien_speed_factor *=self.speed_scale

 self.alien_points = int(self.alien_points*self.score_speed)
 print(self.alien_points) #输出打印看点数是否增加

#2.函数区
#1)定义一个屏幕
def update_screen(setting_1,screen,stats,score_b,ship,aliens,bullets,my_button):
 # 每次循环时绘制屏幕
 screen.fill(setting_1.background)
 #在飞船和外星人后面重新绘制子弹
 #screen.blit(setting_1.background_image,(0,0))
 for bullet in bullets.sprites():
 bullet.draw_bullet()
 #显示得分
 score_b.show_score()
 #让最近绘制的屏幕可见
 ship.ship_1()
 aliens.draw(screen)
 #如果屏幕属于非活跃状态,就绘制play按钮
 if not stats.game_active:
 my_button.draw_button()
 #让最近绘制的屏幕可见
 pygame.display.flip()
# 主函数
def run_deploy():
 pygame.init() #初始化
 setting_1 = Settings() #Settings类实例化
 screen = pygame.display.set_mode((setting_1.screen_width,setting_1.screen_height))
 pygame.display.set_caption("飞机大战")
 #创建Play按钮
 my_button = Button(setting_1,screen,"Play")
 #创建一艘飞船
 ship = Ship(setting_1.ship_speed,screen)
 # 创建一个存储子弹的编组
 bullets = Group()
 #创建一个外星人编组
 aliens = Group()
 #创建存储游戏统计信息的实例
 stats = Game_stats(setting_1)
 #创建记分牌
 score_b = Scoreboard(setting_1,screen,stats)

 #开始游戏
 while True:
 events(setting_1,screen,stats,score_b,my_button,ship,aliens,bullets) #事件监测
 if stats.game_active:
  #当游戏为活跃状态是,更新游戏元素
  ship.moving_1()  #飞船移动
  update_bullet(setting_1,screen,stats,score_b,ship,aliens,bullets) #更新子弹并删除子弹
  update_aliens(setting_1,screen,stats,score_b,ship,aliens,bullets)

 update_screen(setting_1,screen,stats,score_b,ship,aliens,bullets,my_button) #更新屏幕

#2)定义一个飞船
class Ship(Sprite):
 def __init__(self,setting_1,screen):
 #初始化飞船,并设置其起始位置
 super(Ship,self).__init__()
 self.screen = screen
 self.ship_speed_setting = setting_1

 #加载飞船并获取外接矩阵
 self.image = pygame.image.load("../image001/ship.bmp")
 self.rect = self.image.get_rect()
 self.screen_rect = screen.get_rect()
 self.center = float(self.rect.centerx)

 #将图片放在底部中央
 self.rect.centerx = self.screen_rect.centerx
 self.rect.bottom = self.screen_rect.bottom

 #移动标志(目的是连续移动)
 self.moving_right_fag = False
 self.moving_left_fag = False

 def moving_1(self):
 """根据移动标志调整飞船位置"""
 if self.moving_right_fag and self.rect.right < self.screen_rect.right: # screen_rect.right 表示的为界面的宽度
  self.rect.centerx += self.ship_speed_setting
 if self.moving_left_fag and self.rect.left > 0:
  self.rect.centerx -= self.ship_speed_setting

 def ship_1(self):
 """指定的位置绘制飞船"""
 self.screen.blit(self.image,self.rect)

 def center_ship(self):
 #飞船居中
 self.center =self.screen_rect.centerx

#3)检测键盘及鼠标响应
def check_keydown_events(event,setting_1,screen,ship,bullets):
 """响应按键"""
 if event.key == pygame.K_RIGHT: #右移
 ship.moving_right_fag = True
 #print(pygame.K_RIGHT)
 elif event.key == pygame.K_LEFT: #左移
 ship.moving_left_fag = True
 elif event.key == pygame.K_SPACE:
 if len(bullets) <= setting_1.bullets_allow: #当子弹编组中子弹个数小于界面上限制的个数时,才会出现新的 子弹
  #创建一颗子弹,将其放入编组中
  new_bullet = Bullet(setting_1,screen,ship)
  bullets.add(new_bullet)

def check_keyup_events(event,ship):
 if event.key ==pygame.K_RIGHT: #右移
 ship.moving_right_fag = False
 elif event.key == pygame.K_LEFT:
 ship.moving_left_fag = False

def events(setting_1,screen,stats,score_b,my_button,ship,aliens,bullets):
 #设置监听鼠标及键盘事件
 for event in pygame.event.get():
  if event.type == pygame.QUIT: #判断没有任何输入的情况下,返回一个空列表
  # print(pygame.QUIT)
  # print(pygame.event)
  sys.exit()

  elif event.type == pygame.KEYDOWN: #判断键盘事件,返回键盘的整数ID,用于识别按键
  check_keydown_events(event,setting_1,screen,ship,bullets)

  elif event.type == pygame.KEYUP:
  check_keyup_events(event,ship)

  elif event.type == pygame.MOUSEBUTTONDOWN: #单击按钮
  mouse_x,mouse_y = pygame.mouse.get_pos()
  check_play_button(setting_1,screen,stats,score_b,my_button,ship,aliens,bullets,mouse_x,mouse_y)


def check_fleet_edgs(setting_1,aliens):
 """检查外星人移动到的边缘,并采取措施"""
 for alien in aliens.sprites():
 if alien.check_edgs():
  change_fleet_direction(setting_1,aliens)
  break

def change_fleet_direction(setting_1,aliens): #采取的措施
 #将整群人下移,并改变方向
 for alien in aliens.sprites():
 alien.rect.y += setting_1.fleet_drop_speed
 setting_1.fleet_direction *= -1

def ship_hit(setting_1,screen,stats,score_b,ship,aliens,bullets):
 """响应被外星人撞到的飞船"""
 #将飞船数减1
 if stats.ship_left > 0:
 print(stats.ship_left)
 #将飞船数减1
 stats.ship_left -= 1

 #更新记分牌
 score_b.prep_ships()

 #清空外星人和子弹列表
 aliens.empty()
 bullets.empty()

 #创建一群新的外星人,并将飞船放在低端中央

 creet_fleet(setting_1,screen,ship,aliens)
 ship.center_ship()
 #暂停1秒
 time.sleep(1)
 else:
 stats.game_active = False
 pygame.mouse.set_visible(True) #显示光标

def check_aliens_bottom(setting_1,screen,stats,score_b,ship,aliens,bullets):
 """检查是否有外星人到达屏幕底部"""
 screen_rect = screen.get_rect()
 for alien in aliens.sprites():
 if alien.rect.bottom >= screen_rect.bottom:
  # print(2*alien.rect.bottom,"###")
  # print(screen_rect.bottom)

  #向飞船撞到一样处理
  ship_hit(setting_1,screen,stats,score_b,ship,aliens,bullets)
  break
def check_play_button(setting_1,screen,stats,score_b,my_button,ship,aliens,bullets,mouse_x,mouse_y):
 """单击按钮开始游戏"""
 button_clicked = my_button.rect.collidepoint(mouse_x,mouse_y)
 if button_clicked and not stats.game_active :
 #重置游戏速度
 setting_1.init_setting()
 pygame.mouse.set_visible(False) #隐藏光标
 #重置游戏统计信息
 stats.reset_stats()
 stats.game_active = True

 #重复记分牌图形
 score_b.prep_score()
 score_b.prep_high_score()
 score_b.prep_level()
 score_b.prep_ships()

 #清空外星人和子弹列表
 aliens.empty()
 bullets.empty()

 #创建外星人群,然后居中
 creet_fleet(setting_1,screen,ship,aliens)
 ship.center_ship()

def check_high_score(stats,score_b):
 """检查是否但是了最高得分"""
 if stats.score > stats.high_score:
 stats.high_score = stats.score
 score_b.prep_high_score()



# 3)定义射击的子弹

class Bullet(Sprite):
 """飞船发射的子弹类"""

 def __init__(self,setting_1,screen,ship):
 """在飞船所处的位置创建一个子弹对象"""
 super(Bullet,self).__init__() # 初始化父类,此处主要初始化的是Sprite类
 self.screen = screen
 #根据pygame的Rect方法绘制子弹矩形 Rect方法跟4个参数 (x,y,d,h)
 self.rect = pygame.Rect(0,0,setting_1.bullet_width,setting_1.bullet_height)
 self.rect.centerx = ship.rect.centerx
 self.rect.top = ship.rect.top
 self.y = float(self.rect.y)

 self.color = setting_1.bullet_color
 self.speed_factor = setting_1.bullet_speed_factor

 def update(self):
 """向上移动子弹"""
 #更新子弹的位置
 self.y -= self.speed_factor
 #更新子弹的rect位置
 self.rect.y = self.y

 def draw_bullet(self):
 """在屏幕上绘制子弹"""
 pygame.draw.rect(self.screen,self.color,self.rect)
#创建一个子弹更新机制
def update_bullet(setting_1,screen,stats,score_b,ship,aliens,bullets):
 #更新子弹位置,并删除子弹
 bullets.update()
 #删除已消失的子弹,原因是由于pygame无法在屏幕外绘制子弹,而实际上是存在的,为了减少的内存的消耗,和对性能的影响
 for bullet in bullets.copy():
  if bullet.rect.bottom <=0:
  bullets.remove(bullet)
 check_bullet_alien_collision(setting_1,screen,stats,score_b,ship,aliens,bullets)

def check_bullet_alien_collision(setting_1,screen,stats,score_b,ship,aliens,bullets):

 #检查是否有子弹击中外星人,如果是,就删除外星人和子弹,直接调用pygame的groupcollide方法
 collsinos = pygame.sprite.groupcollide(bullets,aliens,True,True)
 #击中外星人后记分
 if collsinos:
  for aliens in collsinos.values(): #为了消除一个外星人被两个子弹击中,或者1个子弹击中多个外星人
  stats.score += setting_1.alien_points*len(aliens)
  score_b.prep_score()
  check_high_score(stats,score_b)
 #如果消灭了所有外星人,子弹将全部消失,一群外星人重新出现
 if len(aliens) == 0 :
  #删除现有的子弹,加快游戏节奏
  bullets.empty()
  setting_1.increase_speed()

  #整群外星人消灭完,等级提升1级
  stats.level += 1
  score_b.prep_level()
  creet_fleet(setting_1,screen,ship,aliens)
# 4)定义一个外星人类
class Alien(Sprite):
 """表示单个外星人的类"""
 def __init__(self,setting_1,screen):
 super(Alien,self).__init__() #初始化外星人,并设置其位置
 self.screen = screen
 self.setting_1 = setting_1

 #加载外星人图片,设置rect属性
 self.image=pygame.image.load("../image001/alien.bmp") # 此处变量为image,不要进行变化,如果为images,程序会报错
 self.rect = self.image.get_rect()
 #外星人的初始位置
 self.rect.x = self.rect.width
 self.rect.y = self.rect.height

 #存储外星人的位置
 self.x = float(self.rect.x)
 def blitme(self):
 """绘制外星人"""
 self.screen.blit(self.image,self.rect)
 #检查外星人是否运动到边沿
 def check_edgs(self):
 screen_rect =self.screen.get_rect()
 if self.rect.right >= screen_rect.right:
  return True
 elif self.rect.left <= 0:
  return True

 def update(self):
 """向左移或右移外星人"""
 self.x += (self.setting_1.alien_speed_factor * self.setting_1.fleet_direction) #注意,此处应该乘以左右移动标志,如果传错参数,可能会导致外星人右移后整体消失
 self.rect.x = self.x #更新位置



def get_number_alien_x(setting_1,alien_width): # 计算每一行可容纳的外星人数
 val_spaces_x = setting_1.screen_width -2*alien_width
 num_alien_x = int(val_spaces_x/(2*alien_width))
 return num_alien_x

def get_number_alien_y(setting_1,ship_height,alien_height):
 avl_spaces_y = (setting_1.screen_height-(3*alien_height)-ship_height)
 number_rows = int(avl_spaces_y/(2*alien_height))
 return number_rows

def creat_aliens(setting_1,screen,aliens,alien_number,row_number):
 """创建一个外星人,并放在当前行"""
 alien = Alien(setting_1,screen)
 alien_width = alien.rect.width
 alien.x = alien_width + 2*alien_width*alien_number
 alien.rect.x = alien.x
 alien.rect.y = alien.rect.height + 2*alien.rect.height*row_number
 aliens.add(alien) #注意此处添加的单个外星人,如果添加成aliens,第一行外星人会不显示,可以用print进行检查

def creet_fleet(setting_1,screen,ship,aliens):
 """创建外星人人群"""
 alien = Alien(setting_1,screen)
 number_alien_x = get_number_alien_x(setting_1,alien.rect.width)
 number_rows = get_number_alien_y(setting_1,ship.rect.height,alien.rect.height)
 #创建外星人群
 for row_number in range(number_rows):
 #创建第一行外星人
 for alien_number in range(number_alien_x):
  creat_aliens(setting_1,screen,aliens,alien_number,row_number)

def update_aliens(setting_1,screen,stats,score_d,ship,aliens,bullets):
 check_fleet_edgs(setting_1,aliens)
 aliens.update()

 # 检查外星人是否与飞船碰撞
 if pygame.sprite.spritecollideany(ship,aliens):
 #print("ship hit")
 ship_hit(setting_1,screen,stats,score_d,ship,aliens,bullets)
 #检查是否有外星人到达底部
 check_aliens_bottom(setting_1,screen,stats,score_d,ship,aliens,bullets)

#5)射杀外星人
#6) 消灭所有的外星人后,外星人群在重新生成,当发生碰撞,游戏结束
#7) 限制飞船的个数3
class Game_stats():
 """ 统计游戏信息"""
 def __init__(self,setting_1):
 #初始化
 self.setting_1 =setting_1
 self.reset_stats()
 #游戏活动状态标志,当为负数时,为False
 self.game_active = False
 #定义在__init__,目的是任何情况下都不重置最高分
 self.high_score = 0
 def reset_stats(self):
 """初始化游戏运行期间可能变化的统计信息"""
 self.ship_left = self.setting_1.ship_limit
 self.score = 0
 self.level = 1 # 等级

#8)添加启动按钮和游戏结束时方便启动
class Button():
 """添加游戏启动按钮"""
 def __init__(self,setting_1,screen,msg):
 """初始化按钮属性"""
 self.screen = screen
 self.screen_rect = screen.get_rect()

 #设置按钮的尺寸及其他属性
 self.width,self.height = 200,100
 self.button_color = (228,222,213)
 self.text_color = (255,255,255)
 self.font = pygame.font.SysFont(None,100)

 #创建按钮的rect对象
 self.rect = pygame.Rect(0,250,self.width,self.height)
 self.rect.centerx = self.screen_rect.centerx

 #按钮的标签只需创建一次
 self.prep_msg(msg)
 def prep_msg(self,msg):
 """将msg渲染成图形,然后居中"""
 self.msg_image = self.font.render(msg,True,self.text_color,self.button_color)
 self.msg_image_rect = self.msg_image.get_rect()
 self.msg_image_rect.center = self.rect.center

 def draw_button(self):
 #绘制一个按钮,在绘制文本
 self.screen.fill(self.button_color,self.rect)
 self.screen.blit(self.msg_image,self.msg_image_rect)


#9)记分系统
class Scoreboard():
 """显示得分信息"""

 def __init__(self,setting_1,screen,stats):
 self.screen = screen
 self.screen_rect = screen.get_rect()
 self.setting_1 = setting_1
 self.stats = stats

 #显示的分
 self.text_color = (30,30,30)
 self.font = pygame.font.SysFont(None,48)

 #准备初始得分图形
 self.prep_score()
 self.prep_high_score()
 self.prep_level()
 self.prep_ships()
 def prep_score(self):

 """得分图形渲染"""
 round_score = int(round(self.stats.score,-1)) # round 四舍五入 ,圆整到10、100、1000的整数倍
 score_str = "{:,}".format(round_score) #输出的格式
 self.score_image = self.font.render(score_str,True,self.text_color,self.setting_1.background)

 #将得分显示在左上角
 self.score_rect = self.score_image.get_rect()
 self.score_rect.right = self.screen_rect.right - 20
 self.screen_rect.top = 20

 def prep_high_score(self): #显示最高得分
 #将最高分的图形渲染
 high_score = int(round(self.stats.high_score,-1)) # round 四舍五入 ,圆整到10、100、1000的整数倍
 high_score_str = "{:,}".format(high_score) #输出的格式
 self.hight_score_image = self.font.render(high_score_str,True,self.text_color,self.setting_1.background)

 #将得分显示在中间角
 self.high_score_rect = self.hight_score_image.get_rect()
 self.high_score_rect.centerx = self.screen_rect.centerx
 self.high_score_rect.top = self.score_rect.top
 def prep_level(self): #显示等级
 """等级渲染图形"""
 self.level_image = self.font.render(str(self.stats.level),True,self.text_color,self.setting_1.background)

 #将等级显示在得分的下方

 self.level_rect = self.level_image.get_rect()
 self.level_rect.right = self.score_rect.right
 self.level_rect.top = self.score_rect.bottom +10
 def prep_ships(self): #显示剩余的飞船数
  self.ships = Group()
  for ship_number in range(self.stats.ship_left):
  ship = Ship(self.setting_1,self.screen)
  ship.rect.x = 10 + ship_number*ship.rect.width
  ship.rect.y = 10
  self.ships.add(ship)


 def show_score(self):
 """在屏幕上显示得分"""
 self.screen.blit(self.score_image,self.score_rect)
 self.screen.blit(self.hight_score_image,self.high_score_rect)
 self.screen.blit(self.level_image,self.level_rect)
 self.ships.draw(self.screen)

#3.调用区
if __name__ == "__main__":
 run_deploy()

所遇到的有5个坑:

坑1:在绘制子弹的时候,执行代码中报错AttributeError: ‘pygame.Surface' object has no attribute ‘bullet_width',分析了大半天,原因是由于形参传错导致的。一定要在程序中保持形参的一致性。

坑2:我在定义读取飞船和外星人的图片时,定义了images参数,我定义了两个images参数,起初在读取图片时,飞船和外星人都能显示在界面上,而当我对外星人进行事件操作,时,程序发生错误 AttributeError: ‘Alien' object has no attribute ‘image'。纠结了好半天,才发现此处的参数应该为image,修改完之后问题解决。飞船的images下面再说

坑3:在定义的外星人的左右移动时,我误将移动标志fleet_direction写成其他值,导致外星人右移消失不见,由于程序未发生报错,于是我采用了print语句,最后才发现是由于移动标志出错导致的。
self.x += (self.setting_1.alien_speed_factor * self.setting_1.fleet_direction)

坑4:在对外星人碰到飞船和碰到界面底部导致游戏结束时,由于的飞船碰到底部逻辑定义有误,导致在最后一个飞船(定义了3个)触碰到界面底部无法终止游戏。由于程序中也没有报错,只能慢慢分析,最终原因是由于对游戏活动状态判断有误,最终将活动状态有True改为False解决<

坑5:第2坑中提到的飞船images参数,导致画剩余的3个飞船是报错(前面绘制的飞船都没有问题) AttributeError: ‘Ship' object has no attribute 'image

小技巧:在对图片上色值采取时,可以在网上在线提取图片中的色值。

性能:在用图片和背景色作为界面的背景时,图片大大降低了游戏的性能。

注:纸上得来终觉浅,绝知此事要躬行。

加载全部内容

相关教程
猜你喜欢
用户评论