pygame自制射击游戏改进
自己用pygame自制的射击游戏
下面是源代码和相关素材
alienInvasion.py
import sys
import pygame
from settings import Settings
from ship import Ship
from bullet import Bullet
from alien import Alien
class AlienInvasion:
def __init__(self):
#初始化游戏资源
pygame.init()
self.settings = Settings()
self.screen = pygame.display.set_mode((self.settings.screen_width, self.settings.screen_height))
pygame.display.set_caption("Alien Invasion")
self.ship = Ship(self)
self.bullets = pygame.sprite.Group()
self.aliens = pygame.sprite.Group()
self._create_fleet()
def run_game(self):
#开始游戏的主循环
while True:
self._check_events()
self.ship.update()
self._update_bullets()
self._update_aliens()
self._update_screen()
def _check_events(self):
"""响应按键和鼠标事件"""
for event in pygame.event.get():
if event.type == pygame.QUIT:
sys.exit()
elif event.type == pygame.KEYDOWN:
self._check_keydown_events(event)
elif event.type == pygame.KEYUP:
self._check_keyup_events(event)
def _check_keydown_events(self, event):
"""响应按键"""
if event.key == pygame.K_RIGHT:
self.ship.moving_right = True
elif event.key == pygame.K_LEFT:
self.ship.moving_left = True
elif event.key == pygame.K_ESCAPE:
sys.exit()
elif event.key == pygame.K_SPACE:
self._fire_bullet()
def _check_keyup_events(self, event):
"""响应松开"""
if event.key == pygame.K_RIGHT:
self.ship.moving_right = False
elif event.key == pygame.K_LEFT:
self.ship.moving_left = False
def _fire_bullet(self):
"""创建一颗子弹,并将其加入编组bullets中"""
new_bullet = Bullet(self)
self.bullets.add(new_bullet)
def _update_bullets(self):
"""更新子弹的位置,并删除已消失的子弹"""
#更新子弹位置
self.bullets.update()
#删除已消失的子弹
for bullet in self.bullets.copy():
if bullet.rect.bottom <= 0:
self.bullets.remove(bullet)
#检查是否有子弹击中了外星人
#如果是这样,就删除相应的子弹和外星人
collisions = pygame.sprite.groupcollide(self.bullets, self.aliens, True, True)
def _create_fleet(self):
"""创建外星人群"""
#创建一个外星人,并计算一行可以容纳多少外星人
#外星人的间距为外星人宽度
alien = Alien(self)
alien_width, alien_height = alien.rect.size
available_space_x = self.settings.screen_width - (3 * alien_width)
number_aliens_x = available_space_x // (3 * alien_width)
#计算屏幕可容纳多少行外星人
ship_height = self.ship.rect.height
available_space_y = (self.settings.screen_height - (3 * alien_height) - ship_height)
number_rows = available_space_y // (4 * alien_height)
#创建外星人群
for row_number in range(number_rows):
for alien_number in range(number_aliens_x):
self._create_alien(alien_number, row_number)
def _create_alien(self, alien_number, row_number):
"""创建一个外星人并将其加入当前行"""
alien = Alien(self)
alien_width, alien_height = alien.rect.size
alien.x = alien_width + 3 * alien_width * alien_number
alien.rect.x = alien.x
alien.rect.y = alien_height + 3 * alien_height * row_number
self.aliens.add(alien)
def _update_aliens(self):
"""检查是否有外星人位于屏幕边缘,并更新整群外星人的位置"""
self._check_fleet_edges()
self.aliens.update()
def _check_fleet_edges(self):
"""有外星人到达边缘时采取相应的措施"""
for alien in self.aliens.sprites():
if alien.check_edges():
self._change_fleet_direction()
break
def _change_fleet_direction(self):
"""将整群外星人下移,并改变它们的移动方向"""
for alien in self.aliens.sprites():
alien.rect.y += self.settings.fleet_drop_speed
self.settings.fleet_direction *= -1
def _update_screen(self):
"""更新屏幕上的图像,并切换到新屏幕"""
self.screen.fill(self.settings.bg_color)
self.ship.blitme()
for bullet in self.bullets.sprites():
bullet.draw_bullet()
self.aliens.draw(self.screen)
#让最近绘制的屏幕可见
pygame.display.flip()
if __name__ == '__main__':
ai = AlienInvasion()
ai.run_game()
settings.py
class Settings:
def __init__(self):
"""存储游戏的所有设置的类"""
#屏幕设置
self.screen_width = 1200
self.screen_height = 800
self.bg_color = (0, 0, 0)
#飞船设置
self.ship_speed = 1.5
#子弹设置
self.bullet_speed = 1.0
self.bullet_width = 100
self.bullet_height = 15
self.bullet_color = (255, 255, 0)
#外星人设置
self.alien_speed = 1.0
self.fleet_drop_speed = 10
#fleet_direction为1表示向右移,为-1表示向左移
self.fleet_direction = 1
ship.py
import pygame
class Ship:
"""管理飞船的类"""
def __init__(self, ai_game):
"""初始化飞船并设置其初始位置"""
self.screen = ai_game.screen
self.settings = ai_game.settings
self.screen_rect = ai_game.screen.get_rect()
#加载飞船图像并获取其外接矩形
self.image = pygame.image.load('images/ship.png')
self.rect = self.image.get_rect()
#对于每艘新飞船,都将其放在屏幕底部中央
self.rect.midbottom = self.screen_rect.midbottom
#在飞船的属性x中存储小数值
self.x = float(self.rect.x)
#移动标志
self.moving_right = False
self.moving_left = False
def update(self):
"""根据移动标志调整飞船的位置"""
#更新飞船而不是rect对象的x值
if self.moving_right and self.rect.right < self.screen_rect.right:
self.x += self.settings.ship_speed
if self.moving_left and self.rect.left > 0:
self.x -= self.settings.ship_speed
#根据self.x更新rect对象
self.rect.x = self.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_game):
"""在飞船所在的位置创建一个子弹对象"""
super().__init__()
self.screen = ai_game.screen
self.settings = ai_game.settings
self.color = self.settings.bullet_color
#在(0, 0)处创建一个表示子弹的矩形,再设置正确的位置
self.rect = pygame.Rect(0, 0, self.settings.bullet_width, self.settings.bullet_height)
self.rect.midtop = ai_game.ship.rect.midtop
#存储用小数表示的子弹位置
self.y = float(self.rect.y)
def update(self):
"""向上移动子弹"""
#更新表示子弹位置的小数值
self.y -= self.settings.bullet_speed
#更新表示子弹的rect的位置
self.rect.y = self.y
def draw_bullet(self):
"""在屏幕上绘制子弹"""
pygame.draw.rect(self.screen, self.color, self.rect)
alien.py
import pygame
from pygame.sprite import Sprite
class Alien(Sprite):
"""表示单个外星人的类"""
def __init__(self, ai_game):
"""初始化外星人并设置其起始位置"""
super().__init__()
self.screen = ai_game.screen
self.settings = ai_game.settings
#加载外星人图像,并设置其rect属性
self.image = pygame.image.load('images/alien.png')
self.rect = self.image.get_rect()
#每个外星人最初都在屏幕左上角附近
self.rect.x = self.rect.width
self.rect.y = self.rect.height
#存储外星人准确水平位置
self.x = float(self.rect.x)
def check_edges(self):
"""如果外星人位于屏幕边缘,就返回True"""
screen_rect = self.screen.get_rect()
if self.rect.right >= screen_rect.right or self.rect.left <= 0:
return True
def update(self):
"""向右移动外星人"""
self.x += (self.settings.alien_speed * self.settings.fleet_direction)
self.rect.x = self.x
没有子弹图片,子弹是用pygame.draw画的矩形
屏幕刷新有点怪怪的,有知道原因的人吗 您目前的代码已经实现了外星人入侵游戏的基本框架:飞船移动、发射子弹、外星人群创建和移动、子弹与外星人碰撞检测。但还没加入游戏结束、计分、飞船生命、音效等经典元素。下面列出几个实用的改进方向,您可以根据兴趣逐步添加。
[*] 添加游戏结束条件
目前外星人飞到屏幕底部或撞到飞船时没有反应。可以在外星人更新后检测是否越界或与飞船碰撞。
示例:在 _update_aliens 方法中加入
for alien in self.aliens.sprites():
if alien.rect.bottom >= self.settings.screen_height:
self._end_game()
if pygame.sprite.spritecollideany(self.ship, self.aliens):
self._end_game()
_end_game 可以打印信息并退出,或重新开始。
[*] 增加计分系统
消灭一个外星人加多少分,显示在屏幕左上角。需要创建 Scoreboard 类,并在碰撞代码中更新分数。
碰撞处(groupcollide)返回一个字典,根据键(子弹)对应的值(外星人列表长度)累加分数。
[*] 飞船生命 / 重试机制
给飞船初始 3 条命,被外星人撞到或外星人到达底部时减少一条命并重置位置。
需要添加 self.ship_hit() 方法:清空子弹和外星人,飞船回到底部中央,生命数减1,生命归0时结束游戏。
[*] 外星人移动加速
随着游戏进行或每消灭一定数量外星人,提高外星人的移动速度。可以在 Settings 中设置 speedup_scale 并在 _update_bullets 检测到外星人为空时调用 _increase_speed 方法。
[*] 限制子弹数量
防止玩家按住空格一直发射。在 Settings 中添加 bullet_allowed 属性,在 _fire_bullet 中检查 len(self.bullets) < self.settings.bullet_allowed。
[*] 添加游戏状态管理
例如用一个 game_active 标志控制主循环是否暂停。当游戏结束时设为 False,并显示“Game Over”文字,按某个键重新开始。
[*] 优化子弹碰撞与爆炸效果
目前碰撞后子弹和外星人瞬间消失,可以添加短暂动画或粒子效果(用精灵组实现简单的缩放或颜色闪烁)。
[*] 音效
使用 pygame.mixer 添加射击、爆炸、背景音乐。注意提前加载并控制音量。
这些改进不需要大改现有结构,每个功能独立,可以挑选最感兴趣的先实现。如果某个部分需要更详细的代码示例,请告诉我,我会针对性地给您写出具体的 BBCode 格式代码段。
以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
页:
[1]