[alien Invasion]python小游戏阶段总结

2024-08-22 16:44

本文主要是介绍[alien Invasion]python小游戏阶段总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

以后可能还会进行代码重构,以最终版本为准

本篇文章旨在理清程序脉络,方便以后写类似的程序时提供一个习惯的思路

未经允许,禁止转载

实体区

ship.py

import pygame
class Ship():def __init__(self,screen,ai_settings):#储存以便后续使用self.screen=screenself.ai_settings=ai_settingsself.image=pygame.image.load(r'images\ship.bmp')#构建rect主体self.rect=self.image.get_rect()self.screen_rect=screen.get_rect()#放中央self.rect.centerx=self.screen_rect.centerx#位置self.rect.bottom=self.screen_rect.bottom#位置#在飞船的属性center储存,实现高精度  self.center=float(self.rect.centerx)#移动标志self.moving_right=Falseself.moving_left=Falsedef update(self):#根据ship_speed_factor调整center,并处理边界问题if self.moving_right and self.rect.right<self.screen_rect.right:self.center+=self.ai_settings.ship_speed_factorif self.moving_left and self.rect.left>0:self.center-=self.ai_settings.ship_speed_factor#根据self.center更新rect对象self.rect.centerx=self.centerdef blitme(self):#画图  self.screen.blit(self.image,self.rect)

ship主要用来制作在screen中的东西,以后的alien.py和这个代码非常像。

alien.py

import pygame
from pygame.sprite import Sprite
class Alien(Sprite):"""表示单个外星人的类"""def __init__(self,ai_settings,screen):"""初始化外星人并设置其起始位置"""super().__init__()#存储screen和setting以便后续使用self.ai_settings=ai_settingsself.screen=screenself.image=pygame.image.load(r'images\alien.bmp')# 加载外星人图像,并设置其rect属性self.rect=self.image.get_rect()# 每个外星人最初都在屏幕左上角附近self.rect.x = self.rect.widthself.rect.y = self.rect.height# 存储外星人的准确位置self.center=float(self.rect.x)def blitme(self):"""在指定位置绘制外星人"""self.screen.blit(self.image,self.rect)

 bullet.py

import pygame
from pygame.sprite import Sprite
class Bullet(Sprite):"""一个对飞船发射的子弹进行管理的类"""def __init__(self,ai_settings,screen,ship):"""在飞船所处的位置创建一个子弹对象"""#继承的例行公事super().__init__()self.screen = screen# 在(0,0)处创建一个表示子弹的矩形,再设置正确的位置self.rect = pygame.Rect(0,0,ai_settings.bullet_width,ai_settings.bullet_height)#移动到正确位置-使用rectself.rect.centerx = ship.rect.centerx self.rect.top = ship.rect.top #存储用小数表示的子弹位置self.y = float(self.rect.y) #颜色和速度self.color = ai_settings.bullet_color self.speed_factor = ai_settings.bullet_speed_factordef update(self):'''向上移动子弹'''#更新表示子弹的rect位置self.y-=self.speed_factor#更新表示子弹的rect位置self.rect.y=self.ydef draw_bullet(self):pygame.draw.rect(self.screen,self.color,self.rect)

ship,alien,bullet,这些实体在构建的时候大同小异

python是动态语言,不需要在实体的类内部实现随着键盘操作而移动的方法

bullet的向上移动是一个例外,这个方法只负责向上移动,也只需要向上移动,因此不需再键鼠操作的代码块里定义该函数,仅仅在主循环之中应用该函数即可

功能区(函数操作区)

game_functions.py

在实体区里,实体类内定义了一些函数,这些函数在功能区被调用,构成了一些新的函数,而这些新的函数会在主函数文件被直接使用。

可能会觉得,某一些代码放在实体类内更合适(比如fire_bullets或许应该放在Bullet类内)。但是,我们在game_function区写下的所有函数,大部分会(也有一部分被自身调用了)在主函数main.py(本文是alien_invasion.py)直接调用。

为了main.py简洁,方便理解的特性,应该尽可能使用代码重构,最常用的方法是打包为方法(也就是函数),所以,实体区函数,功能区函数,main区函数的关系应该是这样的:

代码重构大大提高了我们理解代码的效率,也使得功能区更加简洁。

import sys
import pygame
from bullet import Bullet
from alien import Alien
def fire_bullets(ship,ai_settings,screen,bullets):if len(bullets)<ai_settings.bullets_allowed:new_bullet=Bullet(ai_settings,screen,ship)bullets.add(new_bullet)def check_keydown_events(event,ship,ai_settings,screen,bullets):'''响应按键'''if event.key==pygame.K_RIGHT:ship.moving_right=Trueelif event.key==pygame.K_LEFT:ship.moving_left=Trueelif event.key==pygame.K_SPACE:fire_bullets(ship,ai_settings,screen,bullets)elif event.key==pygame.K_q:sys.exit()def check_keyup_events(event,ship):'''相应松开'''if event.key==pygame.K_RIGHT:ship.moving_right=Falseelif event.key==pygame.K_LEFT:ship.moving_left=False#几类按键
def check_events(ai_settings,screen,ship,bullets):#check eventsfor event in pygame.event.get():if event.type==pygame.QUIT:sys.exit()       elif event.type==pygame.KEYDOWN:check_keydown_events(event,ship,ai_settings,screen,bullets)           elif event.type==pygame.KEYUP:check_keyup_events(event,ship)def update_screen(ai_settings,screen,ship,bullets,aliens):screen.fill(ai_settings.bg_color)# 绘制子弹for bullet in bullets.sprites():bullet.draw_bullet()ship.blitme()aliens.draw(screen)pygame.display.flip()def update_bullets(bullets):"""更新子弹的位置,并删除已消失的子弹"""# 更新子弹的位置bullets.update()# 删除已消失的子弹for bullet in bullets.copy():if bullet.rect.bottom <= 0:bullets.remove(bullet)

主体区(main)

alien_invasion.py

再次强调,python是动态语言,这意味着我们不可以按照c++的先声明再使用,没有具体类型就不能用这些的思考习惯来理解和书写python代码。python的动态特性使得我们可以再main.py里仅仅使用一个Settings类(见下文)来存储所有实体所需要的初始化数值,而不必去每一个实体里面一个个用变量来初始化。

如前文所述,几乎所有功能区的函数都会在main里调用,我们又不想把所有功能区的函数名字引进来以函数名直接调用来造成(可能的)混淆,我们确实import了全部,但是没有不负责任的import *, 所以我们起了个别名gf,以成员的方式来调用函数。

class Settings和run_game()函数我分成两个思维导图来写。

 

import sys
import pygamefrom ship import Ship
from alien import Alien
import game_functions as gf
from pygame.sprite import Groupclass Settings():def __init__(self):self.screen_width=1600self.screen_height=900 self.bg_color=(240,240,240)#0~255self.ship_speed_factor=1.5#一次移动1.5像素1self.bullet_speed_factor=1.5self.bullet_width=3self.bullet_height=15self.bullet_color=60,60,60#元组self.bullets_allowed=3def run_game():pygame.init()ai_settings=Settings()screen=pygame.display.set_mode((ai_settings.screen_width,ai_settings.screen_height))pygame.display.set_caption('Alien Invasion')#创建一艘飞船ship=Ship(screen,ai_settings)#创建用于存储子弹的编组(Group)创建Group实例,只负责提供容器,不管bullet干啥bullets=Group()aliens=Alien()#实例化一个alienalien=Alien(ai_settings,screen)#以后再说...gf.creat_fleet(ai_settings,screen,aliens)while True:gf.check_events(ai_settings,screen,ship,bullets)                ship.update()gf.update_bullets(bullets)                                  gf.update_screen(ai_settings,screen,ship,bullets,aliens)run_game()

仅仅是阶段总结,以后代码会持续更新知道游戏被做出来,本节项目内容来自:Python编程:从入门到实践 - 埃里克·马瑟斯 - 微信读书本书是一本针对所有层次的Python读者而作的Python入门书。全书分两部分:第一部分介绍用Python编程所必须了解的基本概念,包括matplotlib、NumPy和Pygal等强大的Python库和工具介绍,以及列表、字典、if语句、类、文件与异常、代码测试等内容;第二部分将理论付诸实践,讲解如何开发三个项目,包括简单的Python2D游戏开发,如何利用数据生成交互式的信息图,以及创建和定制简单的Web应用,并帮读者解决常见编程问题和困惑。本书适合对Python感兴趣的任何层次的读者阅读。icon-default.png?t=N7T8https://weread.qq.com/web/reader/19532980715c01921954a54

这篇关于[alien Invasion]python小游戏阶段总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1096829

相关文章

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python正则表达式匹配和替换的操作指南

《Python正则表达式匹配和替换的操作指南》正则表达式是处理文本的强大工具,Python通过re模块提供了完整的正则表达式功能,本文将通过代码示例详细介绍Python中的正则匹配和替换操作,需要的朋... 目录基础语法导入re模块基本元字符常用匹配方法1. re.match() - 从字符串开头匹配2.

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

通过Docker容器部署Python环境的全流程

《通过Docker容器部署Python环境的全流程》在现代化开发流程中,Docker因其轻量化、环境隔离和跨平台一致性的特性,已成为部署Python应用的标准工具,本文将详细演示如何通过Docker容... 目录引言一、docker与python的协同优势二、核心步骤详解三、进阶配置技巧四、生产环境最佳实践

Python一次性将指定版本所有包上传PyPI镜像解决方案

《Python一次性将指定版本所有包上传PyPI镜像解决方案》本文主要介绍了一个安全、完整、可离线部署的解决方案,用于一次性准备指定Python版本的所有包,然后导出到内网环境,感兴趣的小伙伴可以跟随... 目录为什么需要这个方案完整解决方案1. 项目目录结构2. 创建智能下载脚本3. 创建包清单生成脚本4

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Python实现批量CSV转Excel的高性能处理方案

《Python实现批量CSV转Excel的高性能处理方案》在日常办公中,我们经常需要将CSV格式的数据转换为Excel文件,本文将介绍一个基于Python的高性能解决方案,感兴趣的小伙伴可以跟随小编一... 目录一、场景需求二、技术方案三、核心代码四、批量处理方案五、性能优化六、使用示例完整代码七、小结一、

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e