python之pygame模块实现飞机大战完整代码-------可直接运行

Python飞机大战步骤:

1.数据区
2.主界面
3.飞船
4.事件监控及边界
5.外星人
6.记分系统
飞机大战效果图 :
python之pygame模块实现飞机大战完整代码-------可直接运行_第1张图片python之pygame模块实现飞机大战完整代码-------可直接运行_第2张图片
源码:

"""
功能:飞机大战
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’,分析了大半天,原因是由于形参传错导致的。一定要在程序中保持形参的一致性。
python之pygame模块实现飞机大战完整代码-------可直接运行_第3张图片
坑2:我在定义读取飞船和外星人的图片时,定义了images参数,我定义了两个images参数,起初在读取图片时,飞船和外星人都能显示在界面上,而当我对外星人进行事件操作,时,程序发生错误 AttributeError: ‘Alien’ object has no attribute ‘image’。纠结了好半天,才发现此处的参数应该为image,修改完之后问题解决。飞船的images下面再说
python之pygame模块实现飞机大战完整代码-------可直接运行_第4张图片
坑3:在定义的外星人的左右移动时,我误将移动标志fleet_direction写成其他值,导致外星人右移消失不见,由于程序未发生报错,于是我采用了print语句,最后才发现是由于移动标志出错导致的。
self.x += (self.setting_1.alien_speed_factor * self.setting_1.fleet_direction)
坑4:在对外星人碰到飞船和碰到界面底部导致游戏结束时,由于的飞船碰到底部逻辑定义有误,导致在最后一个飞船(定义了3个)触碰到界面底部无法终止游戏。由于程序中也没有报错,只能慢慢分析,最终原因是由于对游戏活动状态判断有误,最终将活动状态有True改为False解决
python之pygame模块实现飞机大战完整代码-------可直接运行_第5张图片
坑5:第2坑中提到的飞船images参数,导致画剩余的3个飞船是报错(前面绘制的飞船都没有问题) AttributeError: ‘Ship’ object has no attribute 'image
python之pygame模块实现飞机大战完整代码-------可直接运行_第6张图片
小技巧:在对图片上色值采取时,可以在网上在线提取图片中的色值。
性能:在用图片和背景色作为界面的背景时,图片大大降低了游戏的性能。

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

你可能感兴趣的:(python之pygame模块实现飞机大战完整代码-------可直接运行)