当前位置:   article > 正文

《Python编程:从入门到实践》项目1:外星人入侵(完整版)_python编程从入门到实践外星人入侵

python编程从入门到实践外星人入侵

本文主要是在第13章的基础上添加了第14章的记分环节。

在第14章,主要是添加了游戏启动前的一个Play按钮、游戏等级的设置、击杀外星人后的得分记录并在屏幕上显示当前得分、最高得分、当前等级以及余下的飞船数量这些使游戏具有更佳的娱乐体验的改善。

《Python编程:从入门到实践》第12章:武装飞船

《Python编程:从入门到实践》第13章:外星人来了

本项目代码展示!!!

1.alien_invasion.py

  1. import sys
  2. import pygame
  3. from time import sleep
  4. from settings import Settings
  5. from ship import Ship
  6. from bullet import Bullet
  7. from alien import Alien
  8. from game_stats import GameStats
  9. from button import Button
  10. from scoreboard import Scoreboard
  11. class AlienInvasion:
  12. """ 管理游戏资源和行为的类 """
  13. def __init__(self):
  14. """ 初始化游戏并创建游戏资源 """
  15. pygame.init()
  16. self.settings = Settings()
  17. self.screen = pygame.display.set_mode((0, 0), pygame.FULLSCREEN)
  18. self.settings.screen_width = self.screen.get_rect().width
  19. self.settings.screen_height = self.screen.get_rect().height
  20. pygame.display.set_caption("Alien Invasion")
  21. # 创建一个用于存储游戏统计信息的实例
  22. # 并创建记分牌
  23. self.stats = GameStats(self)
  24. self.sb = Scoreboard(self)
  25. self.ship = Ship(self)
  26. self.bullets = pygame.sprite.Group()
  27. self.aliens = pygame.sprite.Group()
  28. self._create_fleet()
  29. # 创建Play按钮
  30. self.play_button = Button(self, "Play")
  31. def run_game(self):
  32. """ 开始游戏的主循环 """
  33. while True:
  34. self._check_events()
  35. if self.stats.game_active:
  36. self.ship.update()
  37. self._update_bullets()
  38. self._update_aliens()
  39. self._update_screen()
  40. def _update_bullets(self):
  41. """ 更新子弹的位置并删除消失的子弹 """
  42. # 更新子弹的位置
  43. self.bullets.update()
  44. # 删除消失的子弹
  45. for bullet in self.bullets.copy():
  46. if bullet.rect.bottom <= 0:
  47. # ######## 此处有改动
  48. # if bullet.rect.right >= self.ship.screen_rect.right:
  49. self.bullets.remove(bullet)
  50. self._check_bullet_alien_collisions()
  51. def _check_bullet_alien_collisions(self):
  52. """ 响应子弹和外星人碰撞 """
  53. # 删除发生碰撞的子弹和外星人
  54. collisions = pygame.sprite.groupcollide(self.bullets, self.aliens, True, True)
  55. '''
  56. 函数sprite.groupcollide()将一个编组中每个元素的rect同另一个编组中每个元素的rect及进行比较。
  57. '''
  58. if collisions:
  59. # 将消灭的每个外星人都计入得分
  60. for aliens in collisions.values():
  61. self.stats.score += self.settings.alien_points * len(aliens)
  62. self.sb.prep_score()
  63. self.sb.check_high_score()
  64. if not self.aliens:
  65. # 删除现有的所有子弹并新建一群外星人
  66. self.bullets.empty()
  67. self._create_fleet()
  68. self.settings.increase_speed()
  69. # 提高等级
  70. self.stats.level += 1
  71. self.sb.prep_level()
  72. def _update_aliens(self):
  73. """
  74. 检查是否有外星人位于屏幕边缘,并更新外星人群中所有外星人的位置
  75. """
  76. self._check_fleet_edges()
  77. self.aliens.update()
  78. # 检测外星人和飞船之间的碰撞
  79. if pygame.sprite.spritecollideany(self.ship, self.aliens):
  80. print("Ship hit!!!")
  81. self._ship_hit()
  82. # 检查是否有外星人到达了屏幕底端
  83. self._check_aliens_bottom()
  84. def _ship_hit(self):
  85. """ 响应飞船被外星人撞到 """
  86. if self.stats.ships_left > 0:
  87. # 将ships_left减1并更新记分牌
  88. self.stats.ships_left -= 1
  89. self.sb.prep_ships()
  90. # 清空余下的外星人和子弹
  91. self.aliens.empty()
  92. self.bullets.empty()
  93. # 创建一群新的外星人,并将飞船放到屏幕底端的中央
  94. self._create_fleet()
  95. self.ship.center_ship()
  96. # 暂停
  97. sleep(0.5)
  98. else:
  99. self.stats.game_active = False
  100. pygame.mouse.set_visible(True)
  101. def _check_events(self):
  102. """ 响应按键和鼠标事件 """
  103. for event in pygame.event.get():
  104. if event.type == pygame.QUIT:
  105. sys.exit()
  106. elif event.type == pygame.MOUSEBUTTONDOWN:
  107. mouse_pos = pygame.mouse.get_pos()
  108. self._check_play_button(mouse_pos)
  109. elif event.type == pygame.KEYDOWN: # 按下键
  110. self._check_keydown_events(event)
  111. elif event.type == pygame.KEYUP: # 松开键
  112. self._check_keyup_events(event)
  113. def _check_play_button(self, mouse_pos):
  114. """ 在玩家单击Play按钮时开始新游戏 """
  115. button_clicked = self.play_button.rect.collidepoint(mouse_pos)
  116. if button_clicked and not self.stats.game_active:
  117. # 重置游戏设置
  118. self.settings.initialize_dynamic_settings()
  119. self._start_game()
  120. def _start_game(self):
  121. """ 响应单击Play按钮或按键P时开始游戏 """
  122. # 重置游戏统计信息
  123. self.stats.reset_stats()
  124. self.stats.game_active = True
  125. self.sb.prep_score()
  126. self.sb.prep_level()
  127. self.sb.prep_ships()
  128. # 清空余下的外星人和子弹
  129. self.aliens.empty()
  130. self.bullets.empty()
  131. # 创建一群新的外星人并让飞船居中
  132. self._create_fleet()
  133. self.ship.center_ship()
  134. # 隐藏鼠标光标
  135. pygame.mouse.set_visible(False)
  136. def _check_keydown_events(self, event):
  137. """ 响应按键 """
  138. if event.key == pygame.K_RIGHT: # 按下的键是右箭头键
  139. # 向右移动飞船
  140. self.ship.moving_right = True
  141. elif event.key == pygame.K_LEFT: # 按下的键是左箭头键
  142. # 向左移动飞船
  143. self.ship.moving_left = True
  144. elif event.key == pygame.K_UP:
  145. # 向上移动飞船
  146. self.ship.moving_up = True
  147. elif event.key == pygame.K_DOWN:
  148. # 向下移动飞船
  149. self.ship.moving_down = True
  150. elif event.key == pygame.K_q:
  151. # 按键Q以退出游戏
  152. sys.exit()
  153. elif event.key == pygame.K_p:
  154. # 按键P以开始游戏
  155. if not self.stats.game_active:
  156. self._start_game()
  157. elif event.key == pygame.K_SPACE:
  158. # 按空格键以发射子弹
  159. self._fire_bullet()
  160. def _check_keyup_events(self, event):
  161. """ 响应松开 """
  162. if event.key == pygame.K_RIGHT: # 松开的键是右箭头键
  163. self.ship.moving_right = False
  164. elif event.key == pygame.K_LEFT: # 松开的键是左箭头键
  165. self.ship.moving_left = False
  166. elif event.key == pygame.K_UP: # 松开的键是上箭头键
  167. self.ship.moving_up = False
  168. elif event.key == pygame.K_DOWN: # 松开的键是下箭头键
  169. self.ship.moving_down = False
  170. def _fire_bullet(self):
  171. """ 创建一颗子弹,并将其加入编组bullets中 """
  172. if len(self.bullets) < self.settings.bullets_allowed:
  173. new_bullet = Bullet(self)
  174. self.bullets.add(new_bullet)
  175. def _create_fleet(self):
  176. """ 创建外星人群 """
  177. # 创建一个外星人并计算一行可容纳多少个外星人
  178. # 外星人的间距为外星人宽度
  179. alien = Alien(self)
  180. alien_width, alien_height = alien.rect.size
  181. availiable_space_x = self.settings.screen_width - (2 * alien_width)
  182. number_aliens_x = availiable_space_x // (2 * alien_width) + 1
  183. # 计算屏幕可容纳多少行外星人
  184. ship_height = self.ship.rect.height
  185. availiable_space_y = (self.settings.screen_height - (3 * alien_height) - ship_height)
  186. number_rows = availiable_space_y // (2 * alien_height)
  187. # 创建外星人群
  188. for row_number in range(number_rows):
  189. for alien_number in range(number_aliens_x):
  190. self._create_alien(alien_number, row_number)
  191. def _create_alien(self, alien_number, row_number):
  192. """ 创建一个外星人并将其加入到当前行 """
  193. alien = Alien(self)
  194. alien_width, alien_height = alien.rect.size
  195. alien.x = alien_width + 2 * alien_width * alien_number
  196. alien.rect.x = alien.x
  197. # 这里很重要!!!
  198. # 如果写作:
  199. # alien.rect.x = alien_width + 2 * alien_width * alien_number
  200. # 屏幕只会出现一列外星人
  201. # 原因是(alien_width + 2 * alien_width * alien_number)
  202. # 是计算当前外星人在当前行的位置,然后使用外星人的属性x来设置其rect的位置,这是创建外星人行列的过程,并不能展示外星人移动过程
  203. # alien是类Alien()的实例,在Alien()中alien.x在一直变化,利用函数_update_aliens()更新,以此来展示外星人左右移动
  204. alien.rect.y = alien.rect.height + 2 * alien.rect.height * row_number
  205. self.aliens.add(alien)
  206. def _check_fleet_edges(self):
  207. """ 外星人到达边缘时采取相应的措施 """
  208. for alien in self.aliens.sprites():
  209. if alien.check_edges():
  210. self._change_fleet_firection()
  211. break
  212. def _check_aliens_bottom(self):
  213. """ 检查是否有外星人到达了屏幕底端 """
  214. screen_rect = self.screen.get_rect()
  215. for alien in self.aliens.sprites():
  216. if alien.rect.bottom >= screen_rect.bottom:
  217. # 像飞船被撞到一样处理
  218. self._ship_hit()
  219. break
  220. def _change_fleet_firection(self):
  221. """ 将整群外星人下移,并改变它们方向 """
  222. for alien in self.aliens.sprites():
  223. alien.rect.y += self.settings.fleet_drop_speed
  224. self.settings.fleet_direction *= -1
  225. def _update_screen(self):
  226. """ 更新屏幕上的图像,并切换到新屏幕 """
  227. self.screen.fill(self.settings.bg_color)
  228. self.ship.blitme()
  229. for bullet in self.bullets.sprites():
  230. bullet.draw_bullet()
  231. self.aliens.draw(self.screen)
  232. # 显示得分
  233. self.sb.show_score()
  234. # 如果游戏处于非活动状态,就绘制Play按钮
  235. if not self.stats.game_active:
  236. self.play_button.draw_button()
  237. pygame.display.flip()
  238. if __name__ == '__main__':
  239. # 创建游戏实例并运行游戏
  240. ai = AlienInvasion()
  241. ai.run_game()

2.settings.py

  1. class Settings:
  2. """ 存储游戏《外星人入侵》中所有设置的类 """
  3. def __init__(self):
  4. """ 初始化游戏的静态设置 """
  5. # 屏幕设置
  6. self.screen_width = 1500
  7. self.screen_height = 800
  8. self.bg_color = (227, 227, 227)
  9. # 飞船设置
  10. self.ship_speed = 1.5 # 速度:每次循环将移动1.5像素
  11. self.ship_limit = 3 # 飞船数量限制
  12. # 子弹设置
  13. self.bullet_speed = 3.0
  14. self.bullet_width = 3
  15. self.bullet_height = 15
  16. self.bullet_color = (60, 60, 60) # 子弹颜色
  17. self.bullets_allowed = 10 # 允许在屏幕中的子弹数量
  18. # 外星人设置
  19. # self.alien_speed = 1.0
  20. self.fleet_drop_speed = 10
  21. # # fleet_direction为1表示右移,为-1表示左移
  22. # self.fleet_direction = 1
  23. # 加快游戏节奏的速度
  24. self.speedup_scale = 1.07
  25. # 外星人分数的提高速度
  26. self.score_scale = 1.5
  27. self.initialize_dynamic_settings()
  28. def initialize_dynamic_settings(self):
  29. """ 初始化随游戏进行而变化的设置 """
  30. self.ship_speed = 1.5
  31. self.bullet_speed = 3.0
  32. self.alien_speed = 1.0
  33. # fleet_direction为1表示向右,为-1表示向左
  34. self.fleet_direction = 1
  35. # 记分
  36. self.alien_points = 50
  37. def increase_speed(self):
  38. """ 提高速度设置和外星人击杀分数 """
  39. self.ship_speed *= self.speedup_scale
  40. self.bullet_speed *= self.speedup_scale
  41. self.alien_speed *= self.speedup_scale
  42. self.alien_points = int(self.alien_points * self.score_scale)

3.alien.py

  1. import pygame
  2. from pygame.sprite import Sprite
  3. class Alien(Sprite):
  4. """ 表示单个外星人的类 """
  5. def __init__(self, ai_game):
  6. """ 初始化外星人并设置其起始位置 """
  7. super().__init__()
  8. self.screen = ai_game.screen
  9. self.settings = ai_game.settings
  10. # 加载外星人图像并设置其rect属性
  11. self.image = pygame.image.load('images/alien.bmp')
  12. self.rect = self.image.get_rect()
  13. # 每个外星人最初都在屏幕左上角附近
  14. self.rect.x = self.rect.width
  15. self.rect.y = self.rect.height
  16. # 存储外星人的精确水平位置
  17. self.x = float(self.rect.x)
  18. def check_edges(self):
  19. """ 如果外星人位于屏幕边缘,就返回True """
  20. screen_rect = self.screen.get_rect()
  21. if self.rect.right >= screen_rect.right or self.rect.left <= 0:
  22. return True
  23. def update(self):
  24. """ 向左或向右移动外星人 """
  25. self.x += (self.settings.alien_speed * self.settings.fleet_direction)
  26. self.rect.x = self.x

4.ship.py

  1. import pygame
  2. from pygame.sprite import Sprite
  3. class Ship(Sprite):
  4. """ 管理飞船的类 """
  5. def __init__(self, ai_game):
  6. """ 初始化飞船并设置其初始位置 """
  7. super().__init__()
  8. self.screen = ai_game.screen
  9. self.settings = ai_game.settings
  10. self.screen_rect = ai_game.screen.get_rect()
  11. # 加载飞船图像并获取其外接矩形
  12. self.image = pygame.image.load('images/ship.bmp')
  13. self.rect = self.image.get_rect()
  14. # 对于每艘新飞船,都将其放在屏幕底部的中央
  15. self.rect.midbottom = self.screen_rect.midbottom
  16. # # 使飞船出现在屏幕中央而不是屏幕底部的中央
  17. # self.rect.center = self.screen_rect.center
  18. # # 使飞船出现在屏幕左侧
  19. # self.rect.midleft = self.screen_rect.midleft
  20. # 在飞船的属性x中储存小数值
  21. self.x = float(self.rect.x)
  22. self.y = float(self.rect.y)
  23. # 移动标志
  24. self.moving_right = False
  25. self.moving_left = False
  26. self.moving_up = False
  27. self.moving_down = False
  28. def update(self):
  29. """ 根据移动标志调整飞船的位置 """
  30. # 更新飞船而不是rect对象的x值
  31. if self.moving_right and self.rect.right < self.screen_rect.right:
  32. self.x += self.settings.ship_speed
  33. if self.moving_left and self.rect.left > 0:
  34. self.x -= self.settings.ship_speed
  35. # 更新飞船而不是rect对象的y值
  36. if self.moving_down and self.rect.bottom < self.screen_rect.bottom:
  37. self.y += self.settings.ship_speed
  38. if self.moving_up and self.rect.top > 0:
  39. self.y -= self.settings.ship_speed
  40. # 根据self.x更新rect对象
  41. self.rect.x = self.x
  42. # 根据self.y更新rect对象
  43. self.rect.y = self.y
  44. def center_ship(self):
  45. """ 让飞船在屏幕低端居中 """
  46. self.rect.midbottom = self.screen_rect.midbottom
  47. self.x = float(self.rect.x)
  48. self.y = float(self.rect.y)
  49. def blitme(self):
  50. """ 在指定位置绘制飞船 """
  51. self.screen.blit(self.image, self.rect)

5.bullet.py

  1. import pygame
  2. from pygame.sprite import Sprite
  3. class Bullet(Sprite):
  4. """ 管理飞船所发射子弹的类 """
  5. def __init__(self, ai_game):
  6. """ 在飞船当前位置创建一个子弹位置 """
  7. super().__init__()
  8. self.screen = ai_game.screen
  9. self.settings = ai_game.settings
  10. self.color = self.settings.bullet_color
  11. # 在(0,0)处创建一个表示子弹的矩形,再设置正确的位置
  12. self.rect = pygame.Rect(0, 0, self.settings.bullet_width, self.settings.bullet_height)
  13. self.rect.midtop = ai_game.ship.rect.midtop
  14. # 储存用小数表示的子弹位置
  15. self.y = float(self.rect.y)
  16. # self.x = float(self.rect.x)
  17. def update(self):
  18. """ 向上移动子弹 """
  19. # 更新表示子弹位置的小数值
  20. self.y -= self.settings.bullet_speed
  21. # 更新表示子弹的rect的位置
  22. self.rect.y = self.y
  23. # def update(self):
  24. # """ 向右移动子弹 """
  25. # # 更新表示子弹位置的小数值
  26. # self.x += self.settings.bullet_speed
  27. # # 更新表示子弹的rect的位置
  28. # self.rect.x = self.x
  29. def draw_bullet(self):
  30. """ 在屏幕上绘制子弹 """
  31. pygame.draw.rect(self.screen, self.color, self.rect)

6.game_stats.py

  1. class GameStats:
  2. """ 跟踪游戏的统计信息 """
  3. def __init__(self, ai_game):
  4. """ 初始化统计信息 """
  5. self.settings = ai_game.settings
  6. self.reset_stats()
  7. # 游戏刚启动时处于非活动状态
  8. self.game_active = False
  9. # 任何情况下都不应重置最高得分
  10. self.high_score = 0
  11. def reset_stats(self):
  12. """ 初始化在游戏运行期间可能变化的统计信息 """
  13. self.ships_left = self.settings.ship_limit
  14. self.score = 0
  15. self.level = 1

7.button.py

  1. import pygame.font
  2. class Button:
  3. def __init__(self, ai_game, msg):
  4. """ 初始化按钮的属性 """
  5. self.screen = ai_game.screen
  6. self.screen_rect = self.screen.get_rect()
  7. # 设置按钮的尺寸和其他属性
  8. self.width, self.height = 200, 50
  9. self.button_color = (0, 255, 0)
  10. self.text_color = (255, 255, 255)
  11. self.font = pygame.font.SysFont(None, 48)
  12. # 创建按钮的rect对象,并使其居中
  13. self.rect = pygame.Rect(0, 0, self.width, self.height)
  14. self.rect.center = self.screen_rect.center
  15. # 按钮的标签只需创建一次
  16. self._prep_msg(msg)
  17. def _prep_msg(self, msg):
  18. """ 将msg渲染为图像,并使其在按钮上居中 """
  19. self.msg_image = self.font.render(msg, True, self.text_color, self.button_color)
  20. # 让文本图像在按钮上居中
  21. self.msg_image_rect = self.msg_image.get_rect()
  22. self.msg_image_rect.center = self.rect.center
  23. def draw_button(self):
  24. """ 绘制一个用颜色填充的按钮,再绘制文本 """
  25. self.screen.fill(self.button_color, self.rect)
  26. self.screen.blit(self.msg_image, self.msg_image_rect)

8.scoreboard.py

  1. import pygame.font
  2. from pygame.sprite import Group
  3. from ship import Ship
  4. class Scoreboard:
  5. """ 显示得分信息的类 """
  6. def __init__(self, ai_game):
  7. """ 初始化显示得分涉及的属性 """
  8. self.ai_game = ai_game
  9. self.screen = ai_game.screen
  10. self.screen_rect = ai_game.screen.get_rect()
  11. self.settings = ai_game.settings
  12. self.stats = ai_game.stats
  13. # 显示得分信息时使用的字体设置
  14. self.text_color = (30, 30, 30)
  15. self.font = pygame.font.SysFont(None, 48)
  16. # 准备包含最高得分和当前得分的图像
  17. self.prep_score()
  18. self.prep_high_score()
  19. self.prep_level()
  20. self.prep_ships()
  21. def prep_score(self):
  22. """ 将得分转化为一幅渲染的图像 """
  23. rounded_score = round(self.stats.score, -1)
  24. # round()通常让小数精通到小数点后某一位,其中小数位数是由第二个实参指定的。
  25. score_str = "{:,}".format(rounded_score)
  26. self.score_image = self.font.render(score_str, True, self.text_color, self.settings.bg_color)
  27. # 在屏幕右上角显示得分
  28. self.score_rect = self.score_image.get_rect()
  29. self.score_rect.right = self.screen_rect.right - 20
  30. self.score_rect.top = 20
  31. def prep_high_score(self):
  32. """ 将最高得分转换为渲染的图像 """
  33. high_score = round(self.stats.high_score, -1)
  34. high_score_str = "{:,}".format(high_score)
  35. self.high_score_image = self.font.render(
  36. high_score_str, True, self.text_color, self.settings.bg_color)
  37. # 将最高得分放在屏幕顶部中央
  38. self.high_score_rect = self.high_score_image.get_rect()
  39. self.high_score_rect.centerx = self.screen_rect.centerx
  40. self.high_score_rect.top = self.score_rect.top
  41. def check_high_score(self):
  42. """ 检查是否诞生了新的最高得分 """
  43. if self.stats.score > self.stats.high_score:
  44. self.stats.high_score = self.stats.score
  45. self.prep_high_score()
  46. def prep_level(self):
  47. """ 将等级转换为渲染的图像 """
  48. level_str = str(self.stats.level)
  49. self.level_image = self.font.render(level_str, True, self.text_color, self.settings.bg_color)
  50. # 将等级放在得分下方
  51. self.level_rect = self.level_image.get_rect()
  52. self.level_rect.right = self.score_rect.right
  53. self.level_rect.top = self.score_rect.bottom + 10
  54. def prep_ships(self):
  55. """ 显示还余下多少艘飞船 """
  56. self.ships = Group() # 创建一个空编组用于存储飞船实例
  57. for ship_number in range(self.stats.ships_left):
  58. ship = Ship(self.ai_game)
  59. ship.rect.x = 10 + ship_number * ship.rect.width
  60. ship.rect.y = 10
  61. self.ships.add(ship)
  62. def show_score(self):
  63. """ 在屏幕上绘制得分、等级和余下的飞船数 """
  64. self.screen.blit(self.score_image, self.score_rect) # 显示得分
  65. self.screen.blit(self.high_score_image, self.high_score_rect) # 显示最高得分
  66. self.screen.blit(self.level_image, self.level_rect) # 显示等级
  67. self.ships.draw(self.screen) # 显示余下的飞船

想要获取此书电子版的朋友可以查看下面博文

《Python编程:从入门到实践》第12章:武装飞船_笨小古的博客-CSDN博客_python编程从入门到实践第十二章icon-default.png?t=M4ADhttps://blog.csdn.net/weixin_51995147/article/details/123620736

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/寸_铁/article/detail/785403
推荐阅读
相关标签
  

闽ICP备14008679号