外星人入侵

devtools/2024/11/14 20:10:44/

学习于Python编程从入门到实践(Eric Matthes 著)

整体目录:外星人入侵文件夹是打包后的不必在意

图片和音效都是网上下载的

音效下载网站:Free 游戏爆击中 Sound Effects Download - Pixabay

运行效果:可以上下左右移动,空格开火,空格或点击Play开始游戏,飞船碰到外星人或者外星人到达屏幕底部则失去一条命

下面是各个文件的内容

alien.py

python">import pygame
from pygame.sprite import Spriteclass Alien(Sprite):"""表示单个外星人的类"""def __init__(self, ai_settings, screen):"""初始化外星人并设置其起始位置"""super(Alien, self).__init__()self.screen = screenself.ai_settings = ai_settings# 加载外星人图像,并设置其rect属性# self.image = pygame.image.load('E:\\Project\\alien_invasion\\images\\alien.png')self.image = pygame.image.load('images/alien.png')self.image = pygame.transform.scale(self.image, (45, 45))self.rect = self.image.get_rect()# 每个外星人最初都在屏幕左上角附近self.rect.x = self.rect.widthself.rect.y = self.rect.height# 存储外星人的准确位置self.x = float(self.rect.x)def blitme(self):"""在指定位置绘制外星人"""self.screen.blit(self.image, self.rect)def check_edges(self):"""如果外星人位于屏幕边缘,就返回True"""screen_rect = self.screen.get_rect()if self.rect.right >= screen_rect.right:return Trueelif self.rect.left <= 0:return Truedef update(self):"""向左或向右移动外星人"""self.x += (self.ai_settings.alien_speed_factor *self.ai_settings.fleet_direction)self.rect.x = self.x

alien_invasion.py

python">
import pygame
from pygame.sprite import Group
from settings import Settings
from game_stats import GameStats
from ship import Ship
import game_functions as gf
from button import Button
from scoreboard import Scoreboarddef run_game():# 初始化游戏并创建一个屏幕对象# 初始化pygame、设置和屏幕对象pygame.init()ai_settings = Settings()screen = pygame.display.set_mode((ai_settings.screen_width, ai_settings.screen_height))# screen = pygame.display.set_mode((1200, 800))pygame.display.set_caption("Alien Invasion")# 创建一个用于存储游戏统计信息的实例,并创建记分牌stats = GameStats(ai_settings)sb = Scoreboard(ai_settings, screen, stats)# 创建一个飞船ship = Ship(ai_settings, screen)# 创建一个用于存储子弹的编组bullets = Group()# 创建一个外星人编组aliens = Group()# 创建外星人群gf.create_fleet(ai_settings, screen, ship, aliens)# 创建Play按钮play_button = Button(ai_settings, screen, "Play")# 开始游戏的主循环while True:gf.check_events(ai_settings, screen, stats, sb, play_button, ship, aliens, bullets)if stats.game_active:ship.update()# 对每个bullet调用bullet.update()gf.update_bullets(ai_settings, screen, stats, sb, ship, aliens, bullets)gf.update_aliens(ai_settings, screen, stats, sb, ship, aliens, bullets)# 这个print用来检查是否真的删除子弹,保留此条代码会严重拖慢运行速度# print(len(bullets))gf.update_screen(ai_settings, screen, stats, sb, ship, aliens, bullets, play_button)run_game()

bullet.py

python">import pygame
from pygame.sprite import Spriteclass Bullet(Sprite):"""一个对飞船发射的子弹进行管理的类"""def __init__(self, ai_settings, screen, ship):"""在飞船所处的位置创建一个子弹的对象"""super(Bullet, self).__init__() # Python2.7写法,也可写为下面这种# super().__init__()self.screen = screen# 在(0, 0)处创建一个表示子弹的矩形,再设置正确的位置self.rect = pygame.Rect(0, 0, ai_settings.bullet_width, ai_settings.bullet_height)self.rect.centerx = ship.rect.centerxself.rect.top = ship.rect.top# 存储用小数表示的子弹位置self.y = float(ship.rect.y)self.color = ai_settings.bullet_colorself.speed_factor = ai_settings.bullet_speed_factordef update(self):"""向上移动子弹"""# 更新表示子弹位置的小数值self.y -= self.speed_factor# 更新表示子弹的rect的位置self.rect.y = self.ydef draw_bullet(self):"""在屏幕上绘制子弹"""pygame.draw.rect(self.screen, self.color, self.rect)

button.py

python">import pygame.fontclass Button():def __init__(self, ai_settings, screen, msg):"""初始化按钮属性"""self.screen = screenself.screen_rect = screen.get_rect()# 设置按钮的尺寸和其他属性self.width, self.height = 200, 50self.button_color = (0, 255, 0)self.text_color = (255, 255, 255)self.font = pygame.font.SysFont(None, 48)# 创建按钮的rect属性,并使其居中self.rect = pygame.Rect(0, 0, self.width, self.height)self.rect.center = self.screen_rect.center# 按钮的标签只需要创建一次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.centerdef draw_button(self):# 绘制一个用颜色填充的按钮,再绘制文本self.screen.fill(self.button_color, self.rect)self.screen.blit(self.msg_image, self.msg_image_rect)

game_function.py

python">import sys
import pygame
# 设置 mixer 支持 MP3 格式
pygame.mixer.pre_init(44100, -16, 2, 2048)  # 44100是采样率,-16是采样位数,2是声道数,2048是缓冲区大小
pygame.init()
pygame.mixer.init()from bullet import Bullet
from alien import Alien
from time import sleep# 加载音效
# fire_sound = pygame.mixer.Sound('E:\Project\\alien_invasion\\sounds\\fire.mp3')
fire_sound = pygame.mixer.Sound('sounds/fire.mp3')
# hit_sound = pygame.mixer.Sound('E:\Project\\alien_invasion\\sounds\\hit.mp3')
hit_sound = pygame.mixer.Sound('sounds/hit.mp3')
hit_sound.set_volume(0.2)def check_keydown_events(event, ai_settings, screen, stats, sb, ship, aliens, bullets):"""响应按键"""if event.key == pygame.K_RIGHT:# 向右移动飞船ship.moving_right = Trueelif event.key == pygame.K_LEFT:# 向左移动飞船ship.moving_left = Trueelif event.key == pygame.K_UP:# 向上移动飞船ship.moving_up = Trueelif event.key == pygame.K_DOWN:# 向下移动飞船ship.moving_down = Trueelif event.key == pygame.K_SPACE:# 创建一颗子弹,并将其加入到编组bullets中if stats.game_active:fire_bullet(ai_settings, screen, ship, bullets)else:start_game(ai_settings, screen, stats, sb, ship, aliens, bullets)elif event.key == pygame.K_ESCAPE:sys.exit()def fire_bullet(ai_settings, screen, ship, bullets):"""如果还没有到达限制就发射一颗子弹"""if len(bullets) < ai_settings.bullets_allowed:new_bullet = Bullet(ai_settings, screen, ship)fire_sound.play()bullets.add(new_bullet)def check_keyup_events(event, ship):"""松开按键"""if event.key == pygame.K_RIGHT:ship.moving_right = Falseelif event.key == pygame.K_LEFT:ship.moving_left = Falseelif event.key == pygame.K_UP:ship.moving_up = Falseelif event.key == pygame.K_DOWN:ship.moving_down = Falsedef check_events(ai_settings, screen, stats, sb, play_button, ship, aliens, bullets):"""响应按键和鼠标事件"""for event in pygame.event.get():if event.type == pygame.QUIT:sys.exit()elif event.type == pygame.KEYDOWN:check_keydown_events(event, ai_settings, screen, stats, sb, ship, aliens, 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(ai_settings, screen, stats, sb, play_button, ship, aliens, bullets, mouse_x, mouse_y)def check_play_button(ai_settings, screen, stats, sb, play_button, ship, aliens, bullets,mouse_x, mouse_y):"""再玩家单机Play按钮时开始游戏"""button_clicked = play_button.rect.collidepoint(mouse_x, mouse_y)if button_clicked and not stats.game_active:# 重制游戏设置ai_settings.initialize_dynamic_settings()start_game(ai_settings, screen, stats, sb, ship, aliens, bullets)def start_game(ai_settings, screen, stats, sb, ship, aliens, bullets):# 隐藏光标pygame.mouse.set_visible(False)# 重置游戏统计信息stats.reset_stats()stats.game_active = True# 重置记分牌图像sb.prep_score()sb.prep_high_score()sb.prep_level()sb.prep_ships()# 清空外星人列表和子弹列表aliens.empty()bullets.empty()# 创建一群新的外星人,并让飞船居中create_fleet(ai_settings, screen, ship, aliens)ship.center_ship()def update_screen(ai_settings, screen, stats, sb, ship, aliens, bullets, play_button):"""更新屏幕上的图像,并切换到新屏幕"""# 每次循环时都重绘屏幕screen.fill(ai_settings.bg_color)# 在飞船和外星人后面重绘所有子弹for bullet in bullets.sprites():bullet.draw_bullet()ship.blitme()aliens.draw(screen)# 显示得分sb.show_score()# 如果游戏处于非活跃状态,就绘制Play按钮if not stats.game_active:play_button.draw_button()# 让最近绘制的屏幕可见pygame.display.flip()def update_bullets(ai_settings, screen, stats, sb, ship, aliens, bullets):"""更新子弹的位置,并删除已消失的子弹"""# 更新子弹位置bullets.update()# 删除已经消失的子弹for bullet in bullets.copy():if bullet.rect.bottom <= 0:bullets.remove(bullet)check_bullet_alien_collision(ai_settings, screen, stats, sb, ship, aliens, bullets)def check_bullet_alien_collision(ai_settings, screen, stats, sb, ship, aliens, bullets):"""响应子弹和外星人的碰撞"""# 删除发生碰撞子弹和外星人# 若第一个True变为False可以让子弹直到顶部才消失并删除所有击中的外星人反之只击中一个子弹也消失collisions = pygame.sprite.groupcollide(bullets, aliens, True, True)if collisions:for aliens in collisions.values():stats.score += ai_settings.alien_points * len(aliens)sb.prep_score()# 播放音效hit_sound.play()check_high_score(stats, sb)# 检查外星人存在情况if len(aliens) == 0:# 删除现有的子弹,加快节奏,并重新创建一群外星人bullets.empty()ai_settings.increase_speed()# 提高等级stats.level += 1sb.prep_level()create_fleet(ai_settings, screen, ship, aliens)def get_number_alien_x(ai_settings, alien_width):"""计算一行可以有多少个外星人"""available_space_x = ai_settings.screen_width - 2 * alien_widthnumber_aliens_x = int(available_space_x / (2 * alien_width))return number_aliens_xdef get_number_rows(ai_settings, ship_height, alien_height):"""计算屏幕可以有多少行外星人"""available_space_y = (ai_settings.screen_height -(3 * alien_height) - ship_height)number_rows = int(available_space_y / (3 * alien_height))return number_rowsdef create_alien(ai_settings, screen, aliens, alien_number, row_number):"""创建一个外星人并将其放在当前行"""alien = Alien(ai_settings, screen)alien_width = alien.rect.widthalien.x = alien_width + 2 * alien_width * alien_numberalien.rect.x = alien.xalien.rect.y = alien.rect.height + 2 * alien.rect.height * row_numberaliens.add(alien)def create_fleet(ai_settings, screen, ship, aliens):"""创建外星人群"""# 创建一个外星人,并计算一行可以容纳多少个外星人# 外星人之间的间距为外星人宽度alien = Alien(ai_settings, screen)number_aliens_x = get_number_alien_x(ai_settings, alien.rect.width)number_rows = get_number_rows(ai_settings, ship.rect.height, alien.rect.height)# 创建外星人群for row_number in range(number_rows):for alien_number in range(number_aliens_x):create_alien(ai_settings, screen, aliens, alien_number, row_number)def check_fleet_edges(ai_settings, aliens):"""有外星人到达边缘时采取相应的措施"""for alien in aliens.sprites():if alien.check_edges():change_fleet_direction(ai_settings, aliens)breakdef change_fleet_direction(ai_settings, aliens):"""将整个外星人下移, 并改变它们的方向"""for alien in aliens.sprites():alien.rect.y += ai_settings.fleet_drop_speedai_settings.fleet_direction *= -1def ship_hit(ai_settings, screen, stats, sb, ship, aliens, bullets):"""响应被外星人撞到的飞船"""if stats.ships_left > 0:# 将ships_left减 1stats.ships_left -= 1# 更新记分牌sb.prep_ships()# 清空外星人列表和子弹列表aliens.empty()bullets.empty()# 创建一群新的外星人,并将飞船放到屏幕底端中央create_fleet(ai_settings, screen, ship, aliens)ship.center_ship()# 暂停sleep(0.5)else:stats.game_active = Falsepygame.mouse.set_visible(True)def check_aliens_bottom(ai_settings, screen, stats, sb, ship, aliens, bullets):"""检查是否有外星人到达屏幕底端"""screen_rect = screen.get_rect()for alien in aliens.sprites():if alien.rect.bottom >= screen_rect.bottom:# 像飞船被撞一样处理ship_hit(ai_settings, screen, stats, sb, ship, aliens, bullets)breakdef update_aliens(ai_settings, screen, stats, sb, ship, aliens, bullets):"""检查是否有外星人位于屏幕边缘,并更新整群外星人的位置"""check_fleet_edges(ai_settings, aliens)aliens.update()# 检查是否有外星人到达底部check_aliens_bottom(ai_settings, screen, stats, sb, ship, aliens, bullets)# 检查外星人和飞船之间的碰撞if pygame.sprite.spritecollideany(ship, aliens):ship_hit(ai_settings, screen, stats, sb, ship, aliens, bullets)def check_high_score(stats, sb):"""检查是否诞生了新的最高分数"""if stats.score > stats.high_score:stats.high_score = stats.scoresb.prep_high_score()

game_stats.py

python">class GameStats():"""跟踪游戏的统计信息"""def __init__(self, ai_settings):"""初始化统计信息"""self.ai_settings = ai_settingsself.reset_stats()# 游戏刚启动时处于非活动状态self.game_active = False# 在任何情况下都不应重置最高分self.high_score = 0def reset_stats(self):"""初始化在游戏运行期间可能变化的统计信息"""self.ships_left = self.ai_settings.ship_limitself.score = 0self.level = 1

scoreboard.py

python">import pygame.font
from pygame.sprite import Groupfrom ship import Shipclass Scoreboard():"""显示得分信息类"""def __init__(self, ai_settings, screen, stats):"""初始化显示得分涉及的属性"""self.screen = screenself.screen_rect = screen.get_rect()self.ai_settings = ai_settingsself.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):"""将得分转换为一幅渲染的图像"""rounded_score = int(round(self.stats.score, -1))score_str = "{:,}".format(rounded_score)self.score_image = self.font.render(score_str, True, self.text_color,self.ai_settings.bg_color)# 将得分放在屏幕的右上角self.score_rect = self.score_image.get_rect()self.score_rect.right = self.screen_rect.right - 20self.score_rect.top = 5def show_score(self):"""在屏幕中显示飞船和得分"""self.screen.blit(self.score_image, self.score_rect)self.screen.blit(self.high_score_image, self.high_score_rect)self.screen.blit(self.level_image, self.level_rect)# 绘制飞船self.ships.draw(self.screen)def prep_high_score(self):"""将最高得分转换为渲染的图像"""high_score = int(round(self.stats.high_score, -1))high_score_str = "{:,}".format(high_score)self.high_score_image = self.font.render(high_score_str, True,self.text_color, self.ai_settings.bg_color)# 将最高得分放在屏幕顶部中央self.high_score_rect = self.high_score_image.get_rect()self.high_score_rect.centerx = self.screen_rect.centerxself.high_score_rect.top = self.score_rect.topdef prep_level(self):"""将等级转换为渲染的图像"""self.level_image = self.font.render(str(self.stats.level), True,self.text_color, self.ai_settings.bg_color)# 将等级放在得下面self.level_rect = self.level_image.get_rect()self.level_rect.right = self.score_rect.rightself.level_rect.top = self.score_rect.bottom + 10def prep_ships(self):"""显示还剩余多少搜飞船"""self.ships = Group()for ship_number in range(self.stats.ships_left):ship = Ship(self.ai_settings, self.screen)ship.rect.x = 10 + ship_number * ship.rect.widthship.rect.y = 10self.ships.add(ship)

settings.py

python">class Settings():"""存储《外星人入侵》的所有设置类"""def __init__(self):"""初始化游戏静态设置"""# 屏幕设置self.screen_width = 1200self.screen_height = 800self.bg_color = (230, 230, 230)# 飞船的设置self.ship_speed_factor = 1.5self.ship_limit = 3# 子弹设置# self.bullet_speed_factor = 3self.bullet_width = 30self.bullet_height = 15self.bullet_color = 60, 60, 60self.bullets_allowed = 3# 外星人设置self.alien_speed_factor = 1# fleet_drop_speed表示触碰到屏幕边缘时下移的速度self.fleet_drop_speed = 5# fleet_direction为 1 表示向右移动,为 -1 表示向左移动self.fleet_direction = 1# 以什么样的速度加快游戏节奏self.speedup_scale = 1.1# 外星人点数提高速度self.score_scale = 1.5self.initialize_dynamic_settings()def initialize_dynamic_settings(self):"""初始化随游戏进行而变化的设置"""self.ship_speed_factor = 1.5self.bullet_speed_factor = 3self.alien_speed_factor = 1# fleet_direction为 1 表示向右移动,为 -1 表示向左移动self.fleet_direction = 1# 记分self.alien_points = 50def increase_speed(self):"""提高速度设置和外星人点数设置"""self.ship_speed_factor *= self.speedup_scaleself.bullet_speed_factor *= self.speedup_scaleself.alien_speed_factor *= self.speedup_scaleself.alien_points = int(self.alien_points * self.score_scale)

ship.py

python">import pygame
from pygame.sprite import Spriteclass Ship(Sprite):def __init__(self, ai_settings, screen):"""初始化飞船并设置其初始位置"""super(Ship, self).__init__()self.screen = screenself.ai_settings = ai_settings# 加载飞船图像并获取其外接矩形# self.image = pygame.image.load('E:\\Project\\alien_invasion\\images\\ship.bmp')self.image = pygame.image.load('images/ship.bmp')# 调整图片大小self.image = pygame.transform.scale(self.image, (35, 35))self.rect = self.image.get_rect()self.screen_rect = screen.get_rect()# 将每艘新飞船放在屏幕的底部中央self.rect.centerx = self.screen_rect.centerxself.rect.bottom = self.screen_rect.bottom# 在飞船的属性center中存储小数值self.centerx = float(self.rect.centerx)self.centery = float(self.rect.centery)# 移动标志self.moving_right = Falseself.moving_left = Falseself.moving_up = Falseself.moving_down = Falsedef update(self):"""根据移动标志的调整飞船的位置"""# 更新飞船的center值,而不是rectif self.moving_right and self.rect.right < self.screen_rect.right:# self.rect.centerx += 1self.centerx += self.ai_settings.ship_speed_factorif self.moving_left and self.rect.left > 0:# self.rect.centerx -= 1self.centerx -= self.ai_settings.ship_speed_factorif self.moving_up and self.rect.top > 0:self.centery -= self.ai_settings.ship_speed_factorif self.moving_down and self.rect.bottom < self.screen_rect.bottom:self.centery += self.ai_settings.ship_speed_factor# 根据self.center更新rect对象self.rect.centerx = self.centerxself.rect.centery = self.centerydef blitme(self):"""在指定位置绘制飞船"""self.screen.blit(self.image, self.rect)def center_ship(self):"""让飞船在屏幕上居中"""# self.centerx = self.screen_rect.centerxself.centerx = self.screen_rect.centerxself.centery = self.screen_rect.bottom - 15


http://www.ppmy.cn/devtools/133991.html

相关文章

2024 - 苹果审核规则小结

苹果审核千奇百怪&#xff0c;能不能过审核&#xff0c;一方面看实力&#xff0c;一方面也看行情和运气。 有些核心的&#xff0c;需要测试非常久的规则我不能说&#xff0c;但是有些花费点时间&#xff0c;消耗个五六个账号就可以测试出来的规则&#xff0c;可以说一说。 本…

uni-app之数据驱动的picker选择器( uni-data-picker)之可以选择到任意级别

背景说明 uni-app 官方的插件市场有数据驱动选择器&#xff0c;可以用作多级分类的场景。本人引入插件后&#xff0c;发现&#xff0c;在h5和微信小程序都只能选择到叶子级。而在给出的官方组件示例中确并非如此。 以选择年级&#xff0c;而不选择班级。然后&#xff0c;想试试…

ubuntu[无桌面]——使用FileZilla连接本地和虚拟机实现文件共享

在虚拟机上跑命令的时候&#xff0c;有时候需要使用到一些在本机上的文件&#xff0c;但是由于安装的Ubuntu是无桌面的&#xff0c;那么怎么去实现将本地文件拖放到虚拟机上捏&#xff0c;这里记录一下 FileZilla的操作&#xff0c;以及一些问题的解决。 &#xff08;1&#xf…

材质(二)——材质参数化,从源材质继承生成不同的材质实例

继承原材质&#xff0c;对外提供参数。 更改调制不同的参数&#xff0c;生成不同的材质实例。 类似于&#xff0c;类的继承。有一个基类Base.继承生成为子类 A_Base,B_Base,C_Base

ArkTS的进阶语法-4(函数补充,正则表达式)

文章目录 ArkTS的进阶语法1. 泛型1.泛型函数2.使用泛型约束3.多个泛型参数4.泛型接口5.泛型类 2.工具类型**1.partial<Type>** 3.空安全1.联合类型设置为空2.非空断言运算符3.空值合并运算符4.可选链 4.模块化1.默认导入与导出2.按需导入与导出3.全部导入 5.定时器1.setT…

TypeScript在现代前端开发中的应用

&#x1f493; 博客主页&#xff1a;瑕疵的CSDN主页 &#x1f4dd; Gitee主页&#xff1a;瑕疵的gitee主页 ⏩ 文章专栏&#xff1a;《热点资讯》 TypeScript在现代前端开发中的应用 TypeScript在现代前端开发中的应用 TypeScript在现代前端开发中的应用 引言 TypeScript 概述…

学术论文写作丨机器学习与深度学习

目录 第一章、ChatGPT-4o使用方法与技巧 第二章、ChatGPT-4o辅助文献检索、总结与分析 第三章、ChatGPT-4o辅助学术论文选题、创新点挖掘与实验方案设计 第四章、ChatGPT-4o辅助学术论文开题与大纲生成 第五章、ChatGPT-4o辅助学术论文写作马拉松活动介绍 第六章、ChatGP…

SQL 外连接

1 外连接 外连接是一种用于结合两个或多个表的方式&#xff0c;返回至少一个表中的所有记录。 左外连接 LEFT JOIN&#xff0c;左表为驱动表&#xff0c;右表为从表。返回驱动表的所有记录以及从表中的匹配记录。如果从表没有匹配&#xff0c;则结果中从表的部分为NULL。 右…