草庐IT

Python学习日记-第二十四天-飞机大战(敌机出场设计)

arizia 2023-08-15 原文

系列文章目录

·使用定时器添加敌机

·设计Enemy 类


前言

昨天因为有事,所以没有及时更新


一、使用定时器添加敌机

  1. 游戏启动后,每隔一秒会出现一架敌机
  2. 每架敌机向屏幕下方飞行,飞行速度各不相同
  3. 每架敌机出现的水平位置也不尽相同

当敌机从屏幕飞出,不会再飞回屏幕中


1.1   定时器

·在Python中,可以使用 pygame.time.set_timer()来添加定时器

·所谓的定时器,就是每隔一段时间,去执行一些动作

  代码: set_timer(eventid, milliseconds) -> None

·set_timer 可以创建一个事件

·可以在游戏循环的事件监听方法中捕获到该事件

·第1个参数 事件代码需要基于常量pygame.USEREVENT来指定

  USEREVENT是一个整数,再增加的事件可以使用 USEREVENT+1 指定,以此类推

·第2个参数是事件触发 间隔 的毫秒值

定时器事件的监听

·通过pygame.event.get()可以获得当前时刻所有的事件列表

·遍历列表并且判断 event.type 是否等于event.id 如果相等,表示定时器事件发生



1.2   定义并监听创建敌机事件

pygame的定时器使用套路非常固定:

  1. 定义定时器常量——eventid
  2. 在初始化方法中,调用set_timer方法设置定时器事件
  3. 在游戏循环中,监听定时器事件

1)定义事件

·在plane_sprites.py的顶部定义 事件常量

 这里虽然pygame 提供了一个 pygame.USEREVENT的常量,但是在我们开发时,还是会针对我们的游戏需求,以及特有的事件,定义一个名字更加方便我们理解的事件常量,会更加的合理,所以我们在这里创建了一个 CREAT_ENEMY_EVENT的常量

 这里设置的时间是1000,是因为单位是以毫秒计算的,1000毫秒=1秒

 再在之前封装好的 event_handler 监听事件方法里面,增加一个判断,最后再控制台观察“敌机出场”是否是间隔一秒出现一次



二、   设计Enemy类

  1.  游戏启动后,每隔一秒会出现一架敌机
  2.  每架敌机向屏幕下方飞行,飞行速度各不相同
  3.  每架敌机出现的水平位置也不尽相同
  4.  当敌机从屏幕飞出,不会再飞回屏幕中

 

·初始化方法

  指定 敌机图片

  随机 敌机的初始位置和初始速度

·重写update方法

  判断是否飞出屏幕,如果是,从精灵族删除

2.1   敌机类的准备

 代码:

class Enemy(GameSprite):
    """敌机精灵"""

    def __init__(self):

        # 1.调用父类方法,创建敌机精灵,同时指定敌机图片
        super().__init__("./images/enemy1.png")

        # 2.指定敌机的初始随机速度

        # 3.指定敌机的初始随机位置
        pass
    def update(self):

        # 1. 调用父类方法,保持垂直方向的飞行
        super().update()

        # 2. 判断是否飞出屏幕,飞出了,就冲精灵组删除敌机
        if self.rect.y >= SCREEN_RECT.height:
            print("飞出屏幕,需要从精灵组删除")


2.2    创建敌机

演练步骤

1.   在__create_sprites,添加敌机精灵组

   ·敌机是定时被创建的,因此在初始化方法中,不需要创建敌机

 

2.   在__event_handler,创建敌机,并且添加到精灵组

   ·调用精灵组的 add 方法可以像精灵族添加精灵

 

3.   在 __update_sprites,让敌机精灵组调用update和draw方法

代码: 

import pygame
from plane_sprites import *


class PlaneGame(object):
    """飞机大战主游戏"""

    def __init__(self):
        print("游戏初始化")

        # 1.创建游戏的窗口
        self.screen = pygame.display.set_mode(SCREEN_RECT.size)
        # 2.创建游戏的时钟
        self.clock = pygame.time.Clock
        # 3.调用私有方法,精灵和精灵组的创建
        self.__create_sprites()

        # 4. 设置定时器事件-创建敌机 1s
        pygame.time.set_timer(CREATE_ENEMY_EVENT, 1000)

    def __create_sprites(self):

        # 创建背景精灵和精灵组
        bg1 = Background()
        bg2 = Background(True)

        self.back_group = pygame.sprite.Group(bg1, bg2)

        # 创建敌机的精灵组
        self.enemy_group = pygame.sprite.Group()

    def start_game(self):
        print("游戏开始...")

        while True:
            # 1.设置刷新帧率
            self.clock().tick(FRAME_PER_SEC)
            # 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:
                print("敌机出场")
                # 创建敌机精灵
                enemy = Enemy()

                # 将敌机精灵添加到敌机精灵组
                self.enemy_group.add(enemy)

    def __check_collide(self):
        pass

    def __update_sprites(self):

        self.back_group.update()
        self.back_group.draw(self.screen)

        self.enemy_group.update()
        self.enemy_group.draw(self.screen)


    @staticmethod
    def __game_over():
        print("游戏结束")

        pygame.quit()
        exit()

if __name__ == '__main__':

    # 创建游戏对象
    game = PlaneGame()

    # 启动游戏
game.start_game()

运行结果:

 屏幕每间隔1秒钟,就会生出一架敌机


2.3   随机敌机位置和速度

1.   导入模块(随机速度)

·在导入模块时,建议 按照以下顺序导入

  • 官方标准模块导入
  • 第三方模块导入
  • 应用程序导入

·修改 plane_sprites.py增加random的导入(random模块 在Python中是数字随机,之前有讲过的,random模块时官方标准模块)

 import random

 

 运行结果:

 敌机的速度都是随机1 - 3了

2.   随机位置

 

敌机的初始位置x,y的值都是要设置的,通过上面的图解可以了解清楚

使用pygame.Rect 提供的bottom属性,在指定敌机初始位置时,会比较方便

·bottom = y + height

·y = bottom - height

操作:

 代码:

self.rect.bottom = 0

max_x = SCREEN_RECT.width - self.rect.width
self.rect.x = random.randint(0, max_x)

运行结果:

 


2.4    移出屏幕销毁敌机(清除内存)

·敌机移出屏幕之后,如果没有碰撞到英雄,敌机的历史使命结束

·需要从敌机组删除,否则会造成内存浪费

检测敌机被销毁

·__del__内置方法会在对象被销毁前调用,在开发中,可以用于判断对象是否被销毁

·判断敌机是否飞出屏幕,如果是,调用kill() 方法从所有组中删除

 代码:

 def update(self):

        # 1. 调用父类方法,保持垂直方向的飞行
        super().update()

        # 2. 判断是否飞出屏幕,飞出了,就冲精灵组删除敌机
        if self.rect.y >= SCREEN_RECT.height:
            print("飞出屏幕,需要从精灵组删除")

            self.kill()


总结

关于敌机出场的设计就已经设计完成,目前所有的代码总结一下:

1.   plane_sprites.py:

import random
import pygame

# 屏幕大小的常量
SCREEN_RECT = pygame.Rect(0, 0, 480, 700)
# 定义刷新帧率
FRAME_PER_SEC = 60
# 创建敌机的定时器常量
CREATE_ENEMY_EVENT = pygame.USEREVENT


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 __init__(self, is_alt=False):

        # 1.调用父类方法实现精灵的创建(images/rect/speed)
        super().__init__("./images/background.png")

        # 2.判断是否是交替图像,如果是,需要设置初始位置
        if is_alt:
            self.rect.y = -self.rect.height

    def update(self):

        # 1.调用父类的方法实现(垂直在屏幕方向向上移动)
        super().update()

        # 2.判断是否移出屏幕,如果移出屏幕,将图像设置到屏幕上方
        if self.rect.y >= SCREEN_RECT.height:
            self.rect.y = -self.rect.height


class Enemy(GameSprite):
    """敌机精灵"""

    def __init__(self):

        # 1.调用父类方法,创建敌机精灵,同时指定敌机图片
        super().__init__("./images/enemy1.png")

        # 2.指定敌机的初始随机速度1 ~ 3
        self.speed = random.randint(1, 3)

        # 3.指定敌机的初始随机位置
        self.rect.bottom = 0

        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:
            print("飞出屏幕,需要从精灵组删除")

            self.kill()

2.   plane_main.py

import pygame
from plane_sprites import *


class PlaneGame(object):
    """飞机大战主游戏"""

    def __init__(self):
        print("游戏初始化")

        # 1.创建游戏的窗口
        self.screen = pygame.display.set_mode(SCREEN_RECT.size)
        # 2.创建游戏的时钟
        self.clock = pygame.time.Clock
        # 3.调用私有方法,精灵和精灵组的创建
        self.__create_sprites()

        # 4. 设置定时器事件-创建敌机 1s
        pygame.time.set_timer(CREATE_ENEMY_EVENT, 1000)

    def __create_sprites(self):

        # 创建背景精灵和精灵组
        bg1 = Background()
        bg2 = Background(True)

        self.back_group = pygame.sprite.Group(bg1, bg2)

        # 创建敌机的精灵组
        self.enemy_group = pygame.sprite.Group()

    def start_game(self):
        print("游戏开始...")

        while True:
            # 1.设置刷新帧率
            self.clock().tick(FRAME_PER_SEC)
            # 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:
                print("敌机出场")
                # 创建敌机精灵
                enemy = Enemy()

                # 将敌机精灵添加到敌机精灵组
                self.enemy_group.add(enemy)

    def __check_collide(self):
        pass

    def __update_sprites(self):

        self.back_group.update()
        self.back_group.draw(self.screen)

        self.enemy_group.update()
        self.enemy_group.draw(self.screen)


    @staticmethod
    def __game_over():
        print("游戏结束")

        pygame.quit()
        exit()

if __name__ == '__main__':

    # 创建游戏对象
    game = PlaneGame()

    # 启动游戏
game.start_game()

目前主程序运行结果:

敌机随机从屏幕上方飞出,速度设置的是(1-3)之间的随机速度,飞出屏幕之后自动销毁 

有关Python学习日记-第二十四天-飞机大战(敌机出场设计)的更多相关文章

  1. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  2. Python 相当于 Perl/Ruby ||= - 2

    这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Pythonconditionalassignmentoperator对于这样一个简单的问题表示歉意,但是谷歌搜索||=并不是很有帮助;)Python中是否有与Ruby和Perl中的||=语句等效的语句?例如:foo="hey"foo||="what"#assignfooifit'sundefined#fooisstill"hey"bar||="yeah"#baris"yeah"另外,类似这样的东西的通用术语是什么?条件分配是我的第一个猜测,但Wikipediapage跟我想的不太一样。

  3. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  4. 华为OD机试用Python实现 -【明明的随机数】 2023Q1A - 2

    华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o

  5. python - 如何读取 MIDI 文件、更改其乐器并将其写回? - 2

    我想解析一个已经存在的.mid文件,改变它的乐器,例如从“acousticgrandpiano”到“violin”,然后将它保存回去或作为另一个.mid文件。根据我在文档中看到的内容,该乐器通过program_change或patch_change指令进行了更改,但我找不到任何在已经存在的MIDI文件中执行此操作的库.他们似乎都只支持从头开始创建的MIDI文件。 最佳答案 MIDIpackage会为您完成此操作,但具体方法取决于midi文件的原始内容。一个MIDI文件由一个或多个音轨组成,每个音轨是十六个channel中任何一个上的

  6. 「Python|Selenium|场景案例」如何定位iframe中的元素? - 2

    本文主要介绍在使用Selenium进行自动化测试或者任务时,对于使用了iframe的页面,如何定位iframe中的元素文章目录场景描述解决方案具体代码场景描述当我们在使用Selenium进行自动化测试的时候,可能会遇到一些界面或者窗体是使用HTML的iframe标签进行承载的。对于iframe中的标签,如果直接查找是无法找到的,会抛出没有找到元素的异常。比如近在咫尺的例子就是,CSDN的登录窗体就是使用的iframe,大家可以尝试通过F12开发者模式查看到的tag_name,class_name,id或者xpath来定位中的页面元素,会抛出NoSuchElementException异常。解决

  7. LC滤波器设计学习笔记(一)滤波电路入门 - 2

    目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称

  8. CAN协议的学习与理解 - 2

    最近在学习CAN,记录一下,也供大家参考交流。推荐几个我觉得很好的CAN学习,本文也是在看了他们的好文之后做的笔记首先是瑞萨的CAN入门,真的通透;秀!靠这篇我竟然2天理解了CAN协议!实战STM32F4CAN!原文链接:https://blog.csdn.net/XiaoXiaoPengBo/article/details/116206252CAN详解(小白教程)原文链接:https://blog.csdn.net/xwwwj/article/details/105372234一篇易懂的CAN通讯协议指南1一篇易懂的CAN通讯协议指南1-知乎(zhihu.com)视频推荐CAN总线个人知识总

  9. 深度学习部署:Windows安装pycocotools报错解决方法 - 2

    深度学习部署:Windows安装pycocotools报错解决方法1.pycocotools库的简介2.pycocotools安装的坑3.解决办法更多Ai资讯:公主号AiCharm本系列是作者在跑一些深度学习实例时,遇到的各种各样的问题及解决办法,希望能够帮助到大家。ERROR:Commanderroredoutwithexitstatus1:'D:\Anaconda3\python.exe'-u-c'importsys,setuptools,tokenize;sys.argv[0]='"'"'C:\\Users\\46653\\AppData\\Local\\Temp\\pip-instal

  10. python ffmpeg 使用 pyav 转换 一组图像 到 视频 - 2

    2022/8/4更新支持加入水印水印必须包含透明图像,并且水印图像大小要等于原图像的大小pythonconvert_image_to_video.py-f30-mwatermark.pngim_dirout.mkv2022/6/21更新让命令行参数更加易用新的命令行使用方法pythonconvert_image_to_video.py-f30im_dirout.mkvFFMPEG命令行转换一组JPG图像到视频时,是将这组图像视为MJPG流。我需要转换一组PNG图像到视频,FFMPEG就不认了。pyav内置了ffmpeg库,不需要系统带有ffmpeg工具因此我使用ffmpeg的python包装p

随机推荐