用Python实现坦克大战游戏 | 干货贴
点击上方“小白学视觉”,选择加"星标"或“置顶”
重磅干货,第一时间送达
《坦克大战》是1985年日本南梦宫Namco游戏公司在任天堂FC平台上,推出的一款多方位平面射击游戏。游戏以坦克战斗及保卫基地为主题,属于策略型联机类。同时也是FC平台上少有的内建关卡编辑器的几个游戏之一,玩家可自己创建独特的关卡,并通过获取一些道具使坦克和基地得到强化。而今天我们就将利用python还原以下坦克大战的制作。

实验前的准备
首先我们使用的Python版本是3.6.5所用到的模块如下:
Pygame模块用来创建游戏整体框架、精灵等基本架构;
OS模块用来加载本地文件(包括音乐,背景、图片等素材)。

精灵类程序
其中精灵类设置作为基本程序框架用来主函数的调用,其中包括子弹类程序、食物类、家类、砖墙树木等障碍物类、坦克类。具体程序布局如下:
其中子弹类程序,首先需要建立bullet.py程序,建立类包括子弹位置、方向、图片加载、子弹速度等基本信息。具体代码如下:
'''子弹'''class Bullet(pygame.sprite.Sprite):def __init__(self, bullet_image_paths, screensize, direction, position, border_len, is_stronger=False, speed=8, **kwargs):pygame.sprite.Sprite.__init__(self)self.bullet_image_paths = bullet_image_pathsself.width, self.height = screensizeself.direction = directionself.position = positionself.image = pygame.image.load(self.bullet_image_paths.get(direction))self.rect = self.image.get_rect()self.rect.center = position# 地图边缘宽度self.border_len = border_len# 是否为加强版子弹(加强版可碎铁墙)self.is_stronger = is_stronger# 子弹速度self.speed = speed'''移动子弹, 若子弹越界, 则返回True, 否则为False'''def move(self):if self.direction == 'up':self.rect = self.rect.move(0, -self.speed)elif self.direction == 'down':self.rect = self.rect.move(0, self.speed)elif self.direction == 'left':self.rect = self.rect.move(-self.speed, 0)elif self.direction == 'right':self.rect = self.rect.move(self.speed, 0)if (self.rect.top < self.border_len) or (self.rect.bottom > self.height) or (self.rect.left < self.border_len) or (self.rect.right > self.width):return Truereturn False
食物奖励类,建立food.py作为坦克吃到食物时增加生命等基本奖励:
'''食物类. 用于获得奖励'''class Foods(pygame.sprite.Sprite):def __init__(self, food_image_paths, screensize, **kwargs):pygame.sprite.Sprite.__init__(self)self.name = random.choice(list(food_image_paths.keys()))self.image = pygame.image.load(food_image_paths.get(self.name))self.rect = self.image.get_rect()self.rect.left, self.rect.top = random.randint(100, screensize[0]-100), random.randint(100, screensize[1]-100)self.exist_time = 1000def update(self):self.exist_time -= 1return True if self.exist_time < 0 else False
坦克家类,建立home.py存储家基本信息(包括是否存活、图片加载、位置尺寸等)。
'''大本营类'''class Home(pygame.sprite.Sprite):def __init__(self, position, imagepaths, **kwargs):pygame.sprite.Sprite.__init__(self)self.imagepaths = imagepathsself.image = pygame.image.load(self.imagepaths[0])self.rect = self.image.get_rect()self.rect.left, self.rect.top = positionself.alive = True'''被摧毁'''def setDead(self):self.image = pygame.image.load(self.imagepaths[1])self.alive = False'''画到屏幕上'''def draw(self, screen):screen.blit(self.image, self.rect)
砖墙等障碍物类,建立scenes.py其中也是主要位置尺寸的布局:
'''砖墙'''class Brick(pygame.sprite.Sprite):def __init__(self, position, imagepath, **kwargs):pygame.sprite.Sprite.__init__(self)self.image = pygame.image.load(imagepath)self.rect = self.image.get_rect()self.rect.left, self.rect.top = position'''铁墙'''class Iron(pygame.sprite.Sprite):def __init__(self, position, imagepath, **kwargs):pygame.sprite.Sprite.__init__(self)self.image = pygame.image.load(imagepath)self.rect = self.image.get_rect()self.rect.left, self.rect.top = position'''冰'''class Ice(pygame.sprite.Sprite):def __init__(self, position, imagepath, **kwargs):pygame.sprite.Sprite.__init__(self)self.image = pygame.Surface((24, 24))for i in range(2):for j in range(2):self.image.blit(pygame.image.load(imagepath), (12*i, 12*j))self.rect = self.image.get_rect()self.rect.left, self.rect.top = position'''河流'''class River(pygame.sprite.Sprite):def __init__(self, position, imagepath, **kwargs):pygame.sprite.Sprite.__init__(self)self.image = pygame.Surface((24, 24))for i in range(2):for j in range(2):self.image.blit(pygame.image.load(imagepath), (12*i, 12*j))self.rect = self.image.get_rect()self.rect.left, self.rect.top = position'''树'''class Tree(pygame.sprite.Sprite):def __init__(self, position, imagepath, **kwargs):pygame.sprite.Sprite.__init__(self)self.image = pygame.Surface((24, 24))for i in range(2):for j in range(2):self.image.blit(pygame.image.load(imagepath), (12*i, 12*j))self.rect = self.image.get_rect()self.rect.left, self.rect.top = position
坦克类,建立tanks.py包括坦克数量名称、初始位置等信息:
'''玩家坦克类'''class PlayerTank(pygame.sprite.Sprite):def __init__(self, name, player_tank_image_paths, position, border_len, screensize, direction='up', bullet_image_paths=None, protected_mask_path=None, boom_image_path=None, **kwargs):pygame.sprite.Sprite.__init__(self)# 玩家1/玩家2self.name = name# 坦克图片路径self.player_tank_image_paths = player_tank_image_paths.get(name)# 地图边缘宽度self.border_len = border_len# 屏幕大小self.screensize = screensize# 初始坦克方向self.init_direction = direction# 初始位置self.init_position = position# 子弹图片self.bullet_image_paths = bullet_image_paths# 保护罩图片路径self.protected_mask = pygame.image.load(protected_mask_path)self.protected_mask_flash_time = 25self.protected_mask_flash_count = 0self.protected_mask_pointer = False# 坦克爆炸图self.boom_image = pygame.image.load(boom_image_path)self.boom_last_time = 5self.booming_flag = Falseself.boom_count = 0# 坦克生命数量self.num_lifes = 3# 重置self.reset()'''移动'''def move(self, direction, scene_elems, player_tanks_group, enemy_tanks_group, home):# 爆炸时无法移动if self.booming_flag:return# 方向不一致先改变方向if self.direction != direction:self.setDirection(direction)self.switch_count = self.switch_timeself.move_cache_count = self.move_cache_time# 移动(使用缓冲)self.move_cache_count += 1if self.move_cache_count < self.move_cache_time:returnself.move_cache_count = 0if self.direction == 'up':speed = (0, -self.speed)elif self.direction == 'down':speed = (0, self.speed)elif self.direction == 'left':speed = (-self.speed, 0)elif self.direction == 'right':speed = (self.speed, 0)rect_ori = self.rectself.rect = self.rect.move(speed)# --碰到场景元素for key, value in scene_elems.items():if key in ['brick_group', 'iron_group', 'river_group']:if pygame.sprite.spritecollide(self, value, False, None):self.rect = rect_orielif key in ['ice_group']:if pygame.sprite.spritecollide(self, value, False, None):self.rect = self.rect.move(speed)# --碰到其他玩家坦克if pygame.sprite.spritecollide(self, player_tanks_group, False, None):self.rect = rect_ori# --碰到敌方坦克if pygame.sprite.spritecollide(self, enemy_tanks_group, False, None):self.rect = rect_ori# --碰到玩家大本营if pygame.sprite.collide_rect(self, home):self.rect = rect_ori# --碰到边界if self.rect.left < self.border_len:self.rect.left = self.border_lenelif self.rect.right > self.screensize[0]-self.border_len:self.rect.right = self.screensize[0] - self.border_lenelif self.rect.top < self.border_len:self.rect.top = self.border_lenelif self.rect.bottom > self.screensize[1]-self.border_len:self.rect.bottom = self.screensize[1] - self.border_len# 为了坦克轮动特效切换图片self.switch_count += 1if self.switch_count > self.switch_time:self.switch_count = 0self.switch_pointer = not self.switch_pointerself.image = self.tank_direction_image.subsurface((48*int(self.switch_pointer), 0), (48, 48))

游戏界面设置
游戏界面设置包括:开始界面设置、结束界面设置和关卡切换界面设置:
其中游戏开始界面包括玩家数的选择和图片音乐的加载:
'''游戏开始界面'''def gameStartInterface(screen, cfg):background_img = pygame.image.load(cfg.OTHER_IMAGE_PATHS.get('background'))color_white = (255, 255, 255)color_red = (255, 0, 0)font = pygame.font.Font(cfg.FONTPATH, cfg.WIDTH//12)logo_img = pygame.image.load(cfg.OTHER_IMAGE_PATHS.get('logo'))logo_img = pygame.transform.scale(logo_img, (446, 70))logo_rect = logo_img.get_rect()logo_rect.centerx, logo_rect.centery = cfg.WIDTH/2, cfg.HEIGHT//4tank_cursor = pygame.image.load(cfg.PLAYER_TANK_IMAGE_PATHS.get('player1')[0]).convert_alpha().subsurface((0, 144), (48, 48))tank_rect = tank_cursor.get_rect()# 玩家数量选择player_render_white = font.render('1 PLAYER', True, color_white)player_render_red = font.render('1 PLAYER', True, color_red)player_rect = player_render_white.get_rect()player_rect.left, player_rect.top = cfg.WIDTH/2.8, cfg.HEIGHT/2.5players_render_white = font.render('2 PLAYERS', True, color_white)players_render_red = font.render('2 PLAYERS', True, color_red)players_rect = players_render_white.get_rect()players_rect.left, players_rect.top = cfg.WIDTH/2.8, cfg.HEIGHT/2# 游戏提示game_tip = font.render('pressto start', True, color_white) game_tip_rect = game_tip.get_rect()game_tip_rect.centerx, game_tip_rect.top = cfg.WIDTH/2, cfg.HEIGHT/1.4game_tip_flash_time = 25game_tip_flash_count = 0game_tip_show_flag = True# 主循环clock = pygame.time.Clock()is_dual_mode = Falsewhile True:for event in pygame.event.get():if event.type == pygame.QUIT:pygame.quit()sys.exit()elif event.type == pygame.KEYDOWN:if event.key == pygame.K_RETURN:return is_dual_modeelif event.key == pygame.K_UP or event.key == pygame.K_DOWN or event.key == pygame.K_w or event.key == pygame.K_s:is_dual_mode = not is_dual_modescreen.blit(background_img, (0, 0))screen.blit(logo_img, logo_rect)game_tip_flash_count += 1if game_tip_flash_count > game_tip_flash_time:game_tip_show_flag = not game_tip_show_flaggame_tip_flash_count = 0if game_tip_show_flag:screen.blit(game_tip, game_tip_rect)if not is_dual_mode:tank_rect.right, tank_rect.top = player_rect.left-10, player_rect.topscreen.blit(tank_cursor, tank_rect)screen.blit(player_render_red, player_rect)screen.blit(players_render_white, players_rect)else:tank_rect.right, tank_rect.top = players_rect.left-10, players_rect.topscreen.blit(tank_cursor, tank_rect)screen.blit(player_render_white, player_rect)screen.blit(players_render_red, players_rect)pygame.display.update()clock.tick(60)
游戏结束界面包括游戏胜利与失败情况判断和是否退出游戏或重新开始的设置:
'''游戏结束界面'''def gameEndIterface(screen, cfg, is_win=True):background_img = pygame.image.load(cfg.OTHER_IMAGE_PATHS.get('background'))color_white = (255, 255, 255)color_red = (255, 0, 0)font = pygame.font.Font(cfg.FONTPATH, cfg.WIDTH//12)# 游戏失败图gameover_img = pygame.image.load(cfg.OTHER_IMAGE_PATHS.get('gameover'))gameover_img = pygame.transform.scale(gameover_img, (150, 75))gameover_img_rect = gameover_img.get_rect()gameover_img_rect.midtop = cfg.WIDTH/2, cfg.HEIGHT/8gameover_flash_time = 25gameover_flash_count = 0gameover_show_flag = True# 游戏胜利与否的提示if is_win:font_render = font.render('Congratulations, You win!', True, color_white)else:font_render = font.render('Sorry, You fail!', True, color_white)font_rect = font_render.get_rect()font_rect.centerx, font_rect.centery = cfg.WIDTH/2, cfg.HEIGHT/3# 用于选择退出或重新开始tank_cursor = pygame.image.load(cfg.PLAYER_TANK_IMAGE_PATHS.get('player1')[0]).convert_alpha().subsurface((0, 144), (48, 48))tank_rect = tank_cursor.get_rect()restart_render_white = font.render('RESTART', True, color_white)restart_render_red = font.render('RESTART', True, color_red)restart_rect = restart_render_white.get_rect()restart_rect.left, restart_rect.top = cfg.WIDTH/2.4, cfg.HEIGHT/2quit_render_white = font.render('QUIT', True, color_white)quit_render_red = font.render('QUIT', True, color_red)quit_rect = quit_render_white.get_rect()quit_rect.left, quit_rect.top = cfg.WIDTH/2.4, cfg.HEIGHT/1.6is_quit_game = False# 主循环clock = pygame.time.Clock()while True:for event in pygame.event.get():if event.type == pygame.QUIT:pygame.quit()sys.exit()elif event.type == pygame.KEYDOWN:if event.key == pygame.K_RETURN:return is_quit_gameelif event.key == pygame.K_UP or event.key == pygame.K_DOWN or event.key == pygame.K_w or event.key == pygame.K_s:is_quit_game = not is_quit_gamescreen.blit(background_img, (0, 0))gameover_flash_count += 1if gameover_flash_count > gameover_flash_time:gameover_show_flag = not gameover_show_flaggameover_flash_count = 0if gameover_show_flag:screen.blit(gameover_img, gameover_img_rect)screen.blit(font_render, font_rect)if not is_quit_game:tank_rect.right, tank_rect.top = restart_rect.left-10, restart_rect.topscreen.blit(tank_cursor, tank_rect)screen.blit(restart_render_red, restart_rect)screen.blit(quit_render_white, quit_rect)else:tank_rect.right, tank_rect.top = quit_rect.left-10, quit_rect.topscreen.blit(tank_cursor, tank_rect)screen.blit(restart_render_white, restart_rect)screen.blit(quit_render_red, quit_rect)pygame.display.update()clock.tick(60)

游戏界面切换主要是利用进度条加载:
'''关卡切换界面'''def switchLevelIterface(screen, cfg, level_next=1):background_img = pygame.image.load(cfg.OTHER_IMAGE_PATHS.get('background'))color_white = (255, 255, 255)color_gray = (192, 192, 192)font = pygame.font.Font(cfg.FONTPATH, cfg.WIDTH//20)logo_img = pygame.image.load(cfg.OTHER_IMAGE_PATHS.get('logo'))logo_img = pygame.transform.scale(logo_img, (446, 70))logo_rect = logo_img.get_rect()logo_rect.centerx, logo_rect.centery = cfg.WIDTH/2, cfg.HEIGHT//4# 游戏加载提示font_render = font.render('Loading game data, You will enter Level-%s' % level_next, True, color_white)font_rect = font_render.get_rect()font_rect.centerx, font_rect.centery = cfg.WIDTH/2, cfg.HEIGHT/2# 游戏加载进度条gamebar = pygame.image.load(cfg.OTHER_IMAGE_PATHS.get('gamebar')).convert_alpha()gamebar_rect = gamebar.get_rect()gamebar_rect.centerx, gamebar_rect.centery = cfg.WIDTH/2, cfg.HEIGHT/1.4tank_cursor = pygame.image.load(cfg.PLAYER_TANK_IMAGE_PATHS.get('player1')[0]).convert_alpha().subsurface((0, 144), (48, 48))tank_rect = tank_cursor.get_rect()tank_rect.left = gamebar_rect.lefttank_rect.centery = gamebar_rect.centery# 加载所需时间load_time_left = gamebar_rect.right - tank_rect.right + 8# 主循环clock = pygame.time.Clock()while True:for event in pygame.event.get():if event.type == pygame.QUIT:pygame.quit()sys.exit()if load_time_left <= 0:returnscreen.blit(background_img, (0, 0))screen.blit(logo_img, logo_rect)screen.blit(font_render, font_rect)screen.blit(gamebar, gamebar_rect)screen.blit(tank_cursor, tank_rect)pygame.draw.rect(screen, color_gray, (gamebar_rect.left+8, gamebar_rect.top+8, tank_rect.left-gamebar_rect.left-8, tank_rect.bottom-gamebar_rect.top-16))tank_rect.left += 1load_time_left -= 1pygame.display.update()clock.tick(60)

完整代码:
https://pan.baidu.com/s/1BUh9M73AAGkZeDN0IEKdKA
提取码:09bl
作者:李秋键
CSDN博客专家,CSDN达人课作者。硕士在读于中国矿业大学,开发有taptap竞赛获奖等。

