昨天的内容里有了运动的子弹,虽然我们只添加了一个子弹,但你可以看到我们需要记录子弹的x,y坐标,每次要更新它的坐标。如果我们想要有多颗子弹,就需要存储多个坐标。那时候处理起来就不显得那么简单,也许我们可以使用两个list,一个专门存储各个子弹的x坐标,另一个专门存储子弹的y坐标,问题似乎变得没那么复杂,写起来会简单一些。但是我们到现在还没有加入过敌机,如果加入了敌机,加入别的东西的设计,我们将需要很多不同的数据的存储。虽然一个思路清晰的程序员可以记住所有的坐标存储在哪个list里,但是这毕竟比较麻烦。那有什么方法呢?

这就需要提到程序语言的数据抽象方法了,也就是面向对象的抽象方法,可以让我们更好地处理数据。


面向对象的关键在于封装,我们来看看应该怎样封装一个子弹。

描述一个子弹,最主要要用的就是它的图片,坐标的位置。它需要的就这么多,还有是一段处理子弹显示的代码。

#定义一个Bullet类,封装子弹的数据和方法
class Bullet(object):
def __init__(self):
self.x = 0
self.y = -100
self.speed = 600
self.image = pygame.image.load(bullet_image_filename).convert_alpha() def move(self,passed_time_second):
if self.y < 0:
mouseX, mouseY = pygame.mouse.get_pos()
self.x = mouseX - self.image.get_width()/2
self.y = mouseY - self.image.get_width()/2
else:
self.y -= self.speed*passed_time_seconds

这段代码的写法和昨天写的完全一样,但是用了面向对象的写法。然后我们将昨天的代码更改一下:

# -*- coding: utf8 -*-

background_image_filename = 'background.png'
mouse_image_filename = 'hero.png'
bullet_image_filename = 'bullet.png'
#指定图像文件名称 import pygame #导入pygame库
from sys import exit #向sys模块借一个exit函数用来退出程序 #定义一个Bullet类,封装子弹的数据和方法
class Bullet(object):
def __init__(self):
self.x = 0
self.y = -100
self.speed = 600
self.image = pygame.image.load(bullet_image_filename).convert_alpha() def move(self, passed_time_second):
if self.y < 0:
mouseX, mouseY = pygame.mouse.get_pos()
self.x = mouseX - self.image.get_width()/2
self.y = mouseY - self.image.get_width()/2
else:
self.y -= self.speed*passed_time_second pygame.init() #初始化pygame,为使用硬件做准备
screen = pygame.display.set_mode((480, 650), 0, 32)
#创建了一个窗口
pygame.display.set_caption("PlaneFight!")
#设置窗口标题
pygame.mouse.set_visible(False) background = pygame.image.load(background_image_filename).convert()
mouse_cursor = pygame.image.load(mouse_image_filename).convert_alpha()
#加载并转换图像 bullet = Bullet()
clock = pygame.time.Clock()
while True:
#游戏主循环 for event in pygame.event.get():
if event.type == pygame.QUIT:
#接收到退出事件后退出程序
pygame.quit()
exit()
time_passed = clock.tick(100)
time_passed_second = time_passed/1000.0
screen.blit(background, (0,0))
#将背景图画上去
x, y = pygame.mouse.get_pos()
#获得鼠标位置
bullet.move(time_passed_second)#移动子弹 x-= mouse_cursor.get_width() / 2
y-= mouse_cursor.get_height() / 2
#计算光标的左上角位置 screen.blit(bullet.image, (bullet.x, bullet.y))
screen.blit(mouse_cursor, (x, y))
#把元素画上去 pygame.display.update()
#刷新一下画面

这样,我们就掌握了子弹的创建方法。


我们用同样的思路去考虑敌机应该怎样创建。

敌机需要封装的数据,主要也是图片,坐标。以及封装一个处理它坐标变化的方法。

那么我们就可以定义敌机的类:

class Enemy(object):#定义一个Enemy类,封装敌机的数据和方法
def __init__(self):
self.x = 200
self.y = -50
self.speed = 200
self.image = pygame.image.load(enemy_image_filename).convert_alpha() def move(self, passed_time_second):
if self.y < 650:
self.y += self.speed*passed_time_second
else:
self.y = -50

然后用相同的方法给它实例化,显示,就可以得到这样的画面:

敌机和子弹一样,我们就将它显示了出来。

但是,这里有个很明显的问题,那就是敌机每次都从上面的中间飞到下面的中间,和真实游戏相差很多,怎么才能让它多变呢?

这里我们可以import另外一个常用的库了,也就是random

from random import randint

在开始加上这一句,引入randint(),这个方法用于生成随机整数,可以给它两个参数,分别作为下界和上界,使用时下界不能比上界大。

然后我们把这个Enemy类改写成这样:

class Enemy(object):#定义一个Enemy类,封装敌机的数据和方法
def restart(self):
self.x = randint(-30,400)
self.y = randint(-100, -50)
self.speed = randint(100,400)
def __init__(self):
self.restart()
self.image = pygame.image.load(enemy_image_filename).convert_alpha() def move(self, passed_time_second):
if self.y < 650:
self.y += self.speed*passed_time_second
else:
self.restart()

这样,敌机的出现速度,出现位置都会发生变化,就显得没有那么呆板了。

整体的代码如下:

# -*- coding: utf8 -*-

background_image_filename = 'background.png'
mouse_image_filename = 'hero.png'
bullet_image_filename = 'bullet.png'
enemy_image_filename = 'enemy.png'
#指定图像文件名称 import pygame #导入pygame库
from sys import exit #向sys模块借一个exit函数用来退出程序
from random import randint #引入随机数 #定义一个Bullet类,封装子弹的数据和方法
class Bullet(object):
def __init__(self):
self.x = 0
self.y = -100
self.speed = 600
self.image = pygame.image.load(bullet_image_filename).convert_alpha() def move(self, passed_time_second):
if self.y < 0:
mouseX, mouseY = pygame.mouse.get_pos()
self.x = mouseX - self.image.get_width()/2
self.y = mouseY - self.image.get_width()/2
else:
self.y -= self.speed*passed_time_second class Enemy(object):#定义一个Enemy类,封装敌机的数据和方法
def restart(self):
self.x = randint(-30,400)
self.y = randint(-100, -50)
self.speed = randint(100,400)
def __init__(self):
self.restart()
self.image = pygame.image.load(enemy_image_filename).convert_alpha() def move(self, passed_time_second):
if self.y < 650:
self.y += self.speed*passed_time_second
else:
self.restart() pygame.init() #初始化pygame,为使用硬件做准备
screen = pygame.display.set_mode((480, 650), 0, 32)
#创建了一个窗口
pygame.display.set_caption("PlaneFight!")
#设置窗口标题
pygame.mouse.set_visible(False) background = pygame.image.load(background_image_filename).convert()
mouse_cursor = pygame.image.load(mouse_image_filename).convert_alpha()
#加载并转换图像 bullet = Bullet()
enemy = Enemy()
clock = pygame.time.Clock()
while True:
#游戏主循环 for event in pygame.event.get():
if event.type == pygame.QUIT:
#接收到退出事件后退出程序
pygame.quit()
exit()
time_passed = clock.tick(100)
time_passed_second = time_passed/1000.0
screen.blit(background, (0,0))
#将背景图画上去
x, y = pygame.mouse.get_pos()
#获得鼠标位置
bullet.move(time_passed_second)#移动子弹
enemy.move(time_passed_second) x-= mouse_cursor.get_width() / 2
y-= mouse_cursor.get_height() / 2
#计算光标的左上角位置 screen.blit(enemy.image, (enemy.x, enemy.y))
screen.blit(bullet.image, (bullet.x, bullet.y))
screen.blit(mouse_cursor, (x, y))
#把各个元素画上去 pygame.display.update()
#刷新一下画面

今天除了将程序方法改成了面向对象以外,基本上也没有多讲什么。

现在,飞机只有一架,子弹只有一发,敌机也只有一架,感觉不出面向对象的威力,但当它数目多起来时,你会觉得面向对象的抽象的确很省事。

明天我就来给出我的多架飞机的方法。


由于我最近事情比较多,这个更新也写的很拖沓,每天只讲了一点点。对于每天在试着做的人来讲是太慢了。不过我每天都会坚持更新,直到把这个游戏搭建起来。

觉得我讲的慢的朋友,有两个方法,一是自己试着查一些资料,二是先不管我的更新,一周以后再看,到时候就可以看到比较多的变化。

最近比较忙,没办法一气呵成,不好意思了。

最新文章

  1. 用flex做垂直居中
  2. 一个神奇的POS -扫描 现场销售 开单打印票据 安卓物联网POS机 手持开单终端机 省时省力 高效准确!!
  3. 使用扫描二维码打开app
  4. Android 数据库管理— — —更新数据
  5. IOS 多线程02-pthread 、 NSThread 、GCD 、NSOperationQueue、NSRunLoop
  6. 建立MySQL的ODBC
  7. Struts2 文件上传
  8. mysql if对数据进行处理 having对数据进行查询 thinkphp中的exp支持更复杂的where查询
  9. ASP.NET MVC 中@html.ActionLink的几种参数格式
  10. ACdream 1064 完美数
  11. NandFlash ECC 校验
  12. Nginx配置了解
  13. VSTO中Word的查找方式
  14. Hadoop记录-hadoop jmx配置
  15. 二进制安装 kubernetes 1.12(四) - 部署 Node 节点组件
  16. 使用Git Subtree在多个项目中共用同一个子项目
  17. centos7使用snmp
  18. 实验-12-JSP简单入门
  19. IdentityServer Token验证
  20. 输出字符串中最长的单词 C# 算法

热门文章

  1. Leetcode题库——28.实现strStr()
  2. 代码上传不到github远程仓库的经历和总结
  3. Objective-C KVC讲解,包你看懂会用
  4. Delphi中使用OLE方法操作Excel
  5. ES6 学习1
  6. hdu 4686 Arc of Dream(矩阵快速幂)
  7. pyCharm最新2019激活码
  8. 【LaTex】随便学学,
  9. [UVA 10635] Prince ans Princess
  10. 基本数据结构 —— 堆以及堆排序(C++实现)