作者 | 李秋键

出品 | AI科技大本营(rgznai100)

《坦克大战》是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('press <Enter> to 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竞赛获奖等。

更多精彩推荐
  • 300亿美元,AMD为什么要买Xilinx?

  • 无限想象空间,用Python玩转3D人体姿态估计

  • 打通语言理论和统计NLP,Transformers/GNNs架构能做到吗?

  • 神经网络其实和人一样懒惰,喜欢走捷径......

  • 华为出售荣耀为不实消息,赵明曾在内部否认;迅雷前CEO陈磊涉嫌职务侵占罪被调查 ;Python 3.9发布|极客头条

  • “一线城市,年薪30万+,我却裸辞回老家”一个前程序员的 10 年职业思考

用Python实现坦克大战游戏 | 干货贴相关推荐

  1. python 游戏开发框架_Python开发 基于python实现坦克大战游戏

    这篇文章主要为大家详细介绍了基于python实现坦克大战游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 本文实例为大家分享了python实现坦克大战游戏的具体代码, ...

  2. PYTHON实现坦克大战游戏

    python开发的坦克大战游戏 import pygame from pygame.sprite import Sprite import sys import time import random ...

  3. python坦克大战游戏_python实现简单坦克大战

    基于对面向对象编程的思想完成简单的坦克大战游戏.主要目的锻炼面相对象编程思想 同样的在使用python进行游戏编写时需要安装pygame模块 安装方法: pycharm安装方式:File --> ...

  4. python坦克大战游戏_Python开发的坦克大战游戏

    python开发的坦克大战游戏importpygamefrom pygame.sprite importSpriteimportsysimporttimeimportrandom SCREEN_WID ...

  5. python趣味编程-坦克大战游戏

    在上一期我们用Python实现了一个简单的计算器的游戏,这一期我们继续使用Python实现一个简单的坦克大战游戏,让我们开始今天的旅程吧~ 简单 2D 游戏 Python 中的简单 2D 游戏是用Py ...

  6. python调用pygame_python使用pygame模块实现坦克大战游戏

    本文实例为大家分享了pygame模块实现坦克大战游戏的具体代码,供大家参考,具体内容如下 首先,第一步,游戏简单素材的准备. 炮弹, 炮弹,坦克移动.音乐-开火素材. 其次,思路整理. 我们需要几个类 ...

  7. python游戏开发keydown_教你用python开发游戏 python之坦克大战上

    #coding:utf-8import pygame,sys,timefrom random import randintfrom pygame.locals import *"" ...

  8. python坦克大战_Life is short,you need Python——Python实现坦克大战(一)

    先展示一下效果 搓搓小手手,坦克大战即将开始--https://www.zhihu.com/video/1140743290784817152 一.游戏引擎的安装 安装方式有两种:1.pip安装 wi ...

  9. 手把手教你用Python实现“坦克大战”,附详细代码!

    小时候玩的"坦克大战",你还记得吗? 满满的回忆 ! 今天,我们使用Python以及强大的第三方库来实现一个简单的坦克大战游戏. 整体效果 环境依赖 python3.7 pygam ...

最新文章

  1. MinGW+Lua环境配置小结
  2. Linux下的USB总线驱动 mouse
  3. MySQL删除同一个字段的多条记录
  4. 【双100%解法】LeetCode 141 【剑指Offer 23】链表中环的入口节点
  5. html表单文本转换,HTML表格中的垂直(旋转)文本
  6. scrapy 爬取百度知道,多spider子一个项目中,使用一个pielines
  7. 2048游戏简单实现
  8. 集合操作retainAll和removeAll
  9. java堆排序思想及代码实现
  10. 各自然带代表植被_十种常见自然带所对应的植被
  11. 从输入URL到页面加载的过程?由一道题完善自己的Web前端知识体系!
  12. 后缀表达式求值和转换(C++)
  13. python基础代码技巧_5行Python代码实现批量打水印技巧,值得收藏|python基础教程|python入门|python教程...
  14. MySQL - Row size too large (> 8126). Changing some columns to TEXT or BLOB
  15. 应届生和往届生,谁更容易考研成功?
  16. 无人机基础知识:多旋翼无人机系统基本组成
  17. 浏览器只有百度不能上网,其他网站上网正常
  18. 数据结构与算法之数组: Leetcode 914. 卡牌分组 (Typescript版)
  19. Centos7安装wps办公软件
  20. 毕设 家庭理财系统论文

热门文章

  1. Ubuntu命令终端查看使用过的命令
  2. 流式大数据处理的三种框架:Storm,Spark和Samza
  3. 基于Netty实现的轻量级分布式服务框架
  4. JSTL标签库的一些基础实例
  5. virtualenv 在windows下的绿化方法
  6. Linux之文件权限管理
  7. 功能演示:戴尔PowerConnect 8024交换机VLAN的创建与删除
  8. 编程小问题系列(2)——为什么WPF里MediaElement等视频控件不起作用
  9. Windows Server 2008 R2 Beta VHD镜像文件发布
  10. java架构设计理解(一)