Python小游戏之 - 飞机大战 !
用Python写的"飞机大战"小游戏
源代码如下:
# coding=utf-8 import random import os import pygame # 用一个常量来存储屏幕的位置和大小,常量用全大写表示 SCREEN_RECT = pygame.Rect(0,0,480,600) # 定义敌机事件定时器常量值 CREATE_ENEMY_EVENT = pygame.USEREVENT # 定义发射子弹定时器常量值(由于USEREVENT这个常量值被敌机事件占用了,加1来改变数值) HERO_FIRE_EVENT = pygame.USEREVENT + 1 # 飞机大战游戏精灵父类 class GameSprite(pygame.sprite.Sprite): """飞机大战游戏精灵父类""" def __init__(self,image_name,speed=1): # 调用父类的初始化方法 super().__init__() # 定义对象的属性(分别是图像,位置,速度) self.image = pygame.image.load(image_name) self.rect = self.image.get_rect() self.speed = speed def update(self): # 在屏幕的垂直方向上移动 self.rect.y += self.speed # 游戏背景精灵子类 class BackGround(GameSprite): """游戏背景精灵子类""" def update(self): # 调用父类的update方法 super().update() if self.rect.y >= self.rect.height: self.rect.y = -self.rect.height # 敌机精灵子类 class Enemy(GameSprite): """敌机精灵子类""" def __init__(self): # 1.调用父类方法 , 创建敌机精灵 , 同时指定敌机图片 super().__init__("./picture/girl.png") # 2.指定敌机的初始随机速度 self.speed = random.randint(1,5) # 3.指定敌机的初始随机位置 self.rect.y = -self.rect.height max_x = SCREEN_RECT.width - self.rect.width self.rect.x = random.randint(0,max_x) def update(self): # 1.调用父类方法 , 保持垂直方向的飞行 super().update() # 2.判断是否飞出屏幕 , 如果是 , 需要从精灵组删除敌机,释放掉内存 if self.rect.y >= SCREEN_RECT.height: self.kill() def __del__(self): pass # 英雄飞机精灵子类 class Hero(GameSprite): """英雄飞机精灵子类""" def __init__(self): # 1.调用父类方法,定义image_name和初始速度参数 super().__init__("./picture/hero.png",0) # 2.设置英雄的初始位置 self.rect.centerx = SCREEN_RECT.centerx self.rect.y = SCREEN_RECT.height - self.rect.height # 3.创建敌机精灵组 self.bullets = pygame.sprite.Group() def update(self): # 英雄在水平方向移动 self.rect.x += self.speed # 控制英雄不能离开屏幕 if self.rect.x < 0: self.rect.x = 0 elif self.rect.x > SCREEN_RECT.width - self.rect.width: self.rect.x = SCREEN_RECT.width - self.rect.width def fire(self): # 创建子弹精灵 bullet = Bullet() # 指定子弹精灵初始位置 bullet.rect.centerx = self.rect.centerx bullet.rect.y = SCREEN_RECT.height - self.rect.height - 50 # 将子弹添加到子弹精灵组 self.bullets.add(bullet) # 子弹精灵子类 class Bullet(GameSprite): """子弹精灵子类""" def __init__(self): super().__init__("./picture/missile.png",-6) def update(self): super().update() # 判断子弹是否飞出屏幕 , 及时删除子弹释放出内存空间 if self.rect.y < 0: self.kill() def __del__(self): pass # 飞机大战主游戏类 class PlaneGame(object): """飞机大战主游戏类""" def __init__(self): # 1.创建游戏窗口属性 self.screen = pygame.display.set_mode(SCREEN_RECT.size) # 2.创建游戏的时钟属性 self.clock = pygame.time.Clock() # 3.调用私有方法 , 创建精灵和精灵组 self.__create_sprites() # 4.设置定时器事件 - 1s = 1000 ms pygame.time.set_timer(CREATE_ENEMY_EVENT,700) pygame.time.set_timer(HERO_FIRE_EVENT,500) def __create_sprites(self): """定义创建精灵和精灵组方法""" # 创建背景精灵和精灵组 bg1 = BackGround("./picture/background.png") bg2 = BackGround("./picture/background.png") bg2.rect.y = -bg2.rect.height self.back_group = pygame.sprite.Group(bg1,bg2) # 创建敌机精灵组 self.enemy_group = pygame.sprite.Group() # 创建英雄飞机的精灵和精灵组 self.hero = Hero() self.hero_group = pygame.sprite.Group(self.hero) def start_game(self): while True: # 1.设置刷新帧率 self.clock.tick(60) # 2.事件监听 self.__event_handler() # 3.碰撞检测 self.__check_collide() # 4.更新/绘制精灵组 self.__update_sprites() # 5.更新屏幕显示 pygame.display.update() # 事件监听(监听定时器常量) def __event_handler(self): for event in pygame.event.get(): # 判断是否退出游戏 if event.type == pygame.QUIT: # 用类名.的方式来调用静态方法 PlaneGame.__game_over() elif event.type == CREATE_ENEMY_EVENT: # 创建敌机精灵对象 enemy = Enemy() # 将敌机精灵添加到敌机精灵组 self.enemy_group.add(enemy) elif event.type == HERO_FIRE_EVENT: self.hero.fire() # 使用键盘提供的方法获取键盘按键 , 得到的是一个按键元组 keys_pressed = pygame.key.get_pressed() if keys_pressed[pygame.K_RIGHT]: self.hero.speed = 4 elif keys_pressed[pygame.K_LEFT]: self.hero.speed = -4 else: self.hero.speed = 0 # 碰撞检测 def __check_collide(self): # 子弹摧毁敌机 pygame.sprite.groupcollide(self.hero.bullets,self.enemy_group,True,True) # 敌机撞毁英雄飞机,返回的是碰撞的敌机列表 enemies_list = pygame.sprite.spritecollide(self.hero,self.enemy_group,True) if len(enemies_list) > 0: # 列表有内容代表发生碰撞了 self.hero.kill() # 释放掉内存 # 退出游戏 PlaneGame.__game_over() # 更新/绘制精灵组 def __update_sprites(self): self.back_group.update() self.back_group.draw(self.screen) self.enemy_group.update() self.enemy_group.draw(self.screen) self.hero_group.update() self.hero_group.draw(self.screen) self.hero.bullets.update() self.hero.bullets.draw(self.screen) @staticmethod def __game_over(): pygame.quit() print("Game Over!") print("\n\n游戏版本: 18.4.11\n游戏名称: 飞机大战\n游戏平台: windows\n 开发者: ChenBin\n") # exit() os.system("pause") if __name__ == "__main__": # 创建游戏对象 game = PlaneGame() # 启动游戏 game.start_game()