Pygame - Python游戏编程入门(3)

前言

  在上一节我们完成了对玩家飞机的基本操作,这一节我们就来创造出敌人了(°∀°)?~目标有三个,第一个是在屏幕上绘制出敌机,第二个是判断子弹是否击中了敌人,第三个是对被击中的敌人作后续的处理。明白方向后就可以开始了!

正片开始~

  1. 绘制敌机

  随机是游戏中一个很重要的元素,不可预测的机制为游戏带来了更丰富的体验。这次我们要在程序中加入随机数,两行代码:  

# 导入random库中的randint函数
from random import randint
# 返回一个整数N, a<=N<=b
N = randint(a, b)

  这样我们就可以使得敌机每次出现的位置变得不可预测了~(??ω??)

  跟之前的风格类似,我们把敌机封装成类,主要是为了能够更方便地使用碰撞检测的功能。

 1 # 敌人类
 2 class Enemy(pygame.sprite.Sprite):
 3     def __init__(self, enemy_surface, enemy_init_pos):
 4         pygame.sprite.Sprite.__init__(self)
 5         self.image = enemy_surface
 6         self.rect = self.image.get_rect()
 7         self.rect.topleft = enemy_init_pos
 8         self.speed = 2
 9
10     def update(self):
11         self.rect.top += self.speed
12         if self.rect.top > SCREEN_HEIGHT:
13             self.kill()

  依然是超出屏幕区域自动销毁对象,最后就是创建敌人对象并在屏幕上绘制出来:

 1 ...
 2 # enemy1图片 **********************************************************
 3 enemy1_surface = shoot_img.subsurface(pygame.Rect(534, 612, 57, 43))
 4 # ********************************************************************
 5 ...
 6
 7 # 事件循环(main loop)
 8 while True:
 9
10     ...
11
12     # 产生敌机 *****************************************************
13     if ticks % 30 == 0:
14         enemy = Enemy(enemy1_surface, [randint(0, SCREEN_WIDTH - enemy1_surface.get_width()), -enemy1_surface.get_height()])
15         enemy_group.add(enemy)
16     # 控制敌机
17     enemy_group.update()
18     # 绘制敌机
19     enemy_group.draw(screen)
20     # ************************************************************
21
22     ...

  导入图片资源当然是必不可少的啦;我们使用ticks控制敌人产生的频率,每30ticks产生一架新敌机,然后将敌机对象加入一个group,统一操作,每一tick更新一次全体enemy的位置。现在绘制的任务就完成啦~看一下效果:

  虽然enemy绘制出来了,但是现在出现了两个问题;第一,子弹无法击中敌人;第二,敌人无法击毁玩家飞机。下面我们先来解决第一个问题。

  2. 我们的子弹击穿了敌人的铠甲!

  说了那么久,终于说到了“碰撞检测”,游戏中的碰撞检测应用范围很广,不过在pygame中,碰撞检测(collide)的机制其实很简单,就是判断sprite1.rect与sprite2.rect是否重叠。那么在pygame.sprite中,我们可以看到一些函数名中包含collide的函数,这些函数一般是用于检测碰撞的,我们可以大体分为sprite与sprite的碰撞检测,sprite与group的碰撞检测,group与group的碰撞检测。回归问题,子弹是一个group,敌人是一个group,那我们在游戏中检测子弹是否击中了敌人很明显需要用group与group的碰撞检测了~

  pygame.sprite.groupcollide()——检测两个group之间所有sprite的碰撞

  groupcollide(group1, group2, dokill1, dokill2, collided = None) -> Sprite_dict

  group1——精灵组1

  group2——精灵组2

  dokill1——是否杀死发生碰撞时group1中的精灵对象

  dokill2——是否杀死发生碰撞时group2中的精灵对象

  collided——可选参数,可自定义一个回调函数,参数为两个精灵对象,用于自定义两个精灵是否发生碰撞,返回bool值;若忽略此参数,则默认碰撞条件为两个精灵的rect发生重叠

  返回一个包含所有group1中与group2发生碰撞的精灵字典(dict)

  

  现在,我们只需要在程序中加入两行代码:

...

# 创建击毁敌人组
enemy1_down_group = pygame.sprite.Group()

# 事件循环(main loop)
while True:

    ...

    # 检测敌机与子弹的碰撞 *******************************************
    enemy1_down_group.add(pygame.sprite.groupcollide(enemy1_group, hero.bullets1, True, True))
      ...

  创建一个包含被击毁的敌人的group,然后在每一tick中检测一次是否发生碰撞,再将被击毁的敌人加入这个group,方便后续对坠毁敌机的动画渲染;这样第二部分也完成啦~

  3. 华丽的坠毁(`?ω?´)

  现在我们已经实现子弹与敌机的碰撞检测了,但凭空消失是在不咋的,我们来一个华丽一点的爆炸!(°∀°)?

  首先导入enemy1爆炸的资源图片

1 enemy1_down_surface = []
2 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(267, 347, 57, 43)))
3 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(873, 697, 57, 43)))
4 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(267, 296, 57, 43)))
5 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(930, 697, 57, 43)))

  然后就是控制爆炸图片切换的速度了,在主循环中加入:

1 for enemy1_down in enemy1_down_group:
2         screen.blit(enemy1_down_surface[enemy1_down.down_index], enemy1_down.rect)
3         if ticks % (ANIMATE_CYCLE//2) == 0:
4             if enemy1_down.down_index < 3:
5                 enemy1_down.down_index += 1
6             else:
7                 enemy1_down_group.remove(enemy1_down)

  当index超出图片下标,判断为爆炸效果演示完毕,销毁坠毁的enemy1精灵。

  这样爆炸效果就出来啦~~

  这一节任务完成!~附上完整代码:

  1 # -*- coding = utf-8 -*-
  2 """
  3 @author: Will Wu
  4 """
  5
  6 import pygame                   # 导入pygame库
  7 from pygame.locals import *     # 导入pygame库中的一些常量
  8 from sys import exit            # 导入sys库中的exit函数
  9 from random import randint
 10
 11 # 定义窗口的分辨率
 12 SCREEN_WIDTH = 480
 13 SCREEN_HEIGHT = 640
 14
 15 # 子弹类
 16 class Bullet(pygame.sprite.Sprite):
 17
 18     def __init__(self, bullet_surface, bullet_init_pos):
 19         pygame.sprite.Sprite.__init__(self)
 20         self.image = bullet_surface
 21         self.rect = self.image.get_rect()
 22         self.rect.topleft = bullet_init_pos
 23         self.speed = 8
 24
 25     # 控制子弹移动
 26     def update(self):
 27         self.rect.top -= self.speed
 28         if self.rect.bottom < 0:
 29             self.kill()
 30
 31
 32 # 玩家类
 33 class Hero(pygame.sprite.Sprite):
 34
 35     def __init__(self, hero_surface, hero_init_pos):
 36         pygame.sprite.Sprite.__init__(self)
 37         self.image = hero_surface
 38         self.rect = self.image.get_rect()
 39         self.rect.topleft = hero_init_pos
 40         self.speed = 6
 41
 42         # 子弹1的Group
 43         self.bullets1 = pygame.sprite.Group()
 44
 45     # 控制射击行为
 46     def single_shoot(self, bullet1_surface):
 47         bullet1 = Bullet(bullet1_surface, self.rect.midtop)
 48         self.bullets1.add(bullet1)
 49
 50     # 控制飞机移动
 51     def move(self, offset):
 52         x = self.rect.left + offset[pygame.K_RIGHT] - offset[pygame.K_LEFT]
 53         y = self.rect.top + offset[pygame.K_DOWN] - offset[pygame.K_UP]
 54         if x < 0:
 55             self.rect.left = 0
 56         elif x > SCREEN_WIDTH - self.rect.width:
 57             self.rect.left = SCREEN_WIDTH - self.rect.width
 58         else:
 59             self.rect.left = x
 60
 61         if y < 0:
 62             self.rect.top = 0
 63         elif y > SCREEN_HEIGHT - self.rect.height:
 64             self.rect.top = SCREEN_HEIGHT - self.rect.height
 65         else:
 66             self.rect.top = y
 67
 68 # 敌人类
 69 class Enemy(pygame.sprite.Sprite):
 70     def __init__(self, enemy_surface, enemy_init_pos):
 71         pygame.sprite.Sprite.__init__(self)
 72         self.image = enemy_surface
 73         self.rect = self.image.get_rect()
 74         self.rect.topleft = enemy_init_pos
 75         self.speed = 2
 76
 77         # 爆炸动画画面索引
 78         self.down_index = 0
 79
 80     def update(self):
 81         self.rect.top += self.speed
 82         if self.rect.top > SCREEN_HEIGHT:
 83             self.kill()
 84
 85 ###########################################################################
 86
 87 # 定义画面帧率
 88 FRAME_RATE = 60
 89
 90 # 定义动画周期(帧数)
 91 ANIMATE_CYCLE = 30
 92
 93 ticks = 0
 94 clock = pygame.time.Clock()
 95 offset = {pygame.K_LEFT:0, pygame.K_RIGHT:0, pygame.K_UP:0, pygame.K_DOWN:0}
 96
 97
 98 # 初始化游戏
 99 pygame.init()                   # 初始化pygame
100 screen = pygame.display.set_mode([SCREEN_WIDTH, SCREEN_HEIGHT])     # 初始化窗口
101 pygame.display.set_caption(‘This is my first pygame-program‘)       # 设置窗口标题
102
103 # 载入背景图
104 background = pygame.image.load(‘resources/image/background.png‘)
105
106 # 载入资源图片
107 shoot_img = pygame.image.load(‘resources/image/shoot.png‘)
108
109 # 用subsurface剪切读入的图片
110 # Hero图片
111 hero_surface = []
112 hero_surface.append(shoot_img.subsurface(pygame.Rect(0, 99, 102, 126)))
113 hero_surface.append(shoot_img.subsurface(pygame.Rect(165, 360, 102, 126)))
114 #hero_surface.append(shoot_img.subsurface(pygame.Rect(165, 234, 102, 126)))
115 #hero_surface.append(shoot_img.subsurface(pygame.Rect(330, 624, 102, 126)))
116 #hero_surface.append(shoot_img.subsurface(pygame.Rect(330, 498, 102, 126)))
117 #hero_surface.append(shoot_img.subsurface(pygame.Rect(432, 624, 102, 126)))
118 hero_pos = [200, 500]
119
120 # bullet1图片
121 bullet1_surface = shoot_img.subsurface(pygame.Rect(1004, 987, 9, 21))
122
123 # enemy1图片 **********************************************************
124 enemy1_surface = shoot_img.subsurface(pygame.Rect(534, 612, 57, 43))
125 enemy1_down_surface = []
126 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(267, 347, 57, 43)))
127 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(873, 697, 57, 43)))
128 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(267, 296, 57, 43)))
129 enemy1_down_surface.append(shoot_img.subsurface(pygame.Rect(930, 697, 57, 43)))
130 # ********************************************************************
131
132 # 创建玩家
133 hero = Hero(hero_surface[0], hero_pos)
134
135 # 创建敌人组
136 enemy1_group = pygame.sprite.Group()
137
138 # 创建击毁敌人组
139 enemy1_down_group = pygame.sprite.Group()
140
141 # 事件循环(main loop)
142 while True:
143
144     # 控制游戏最大帧率
145     clock.tick(FRAME_RATE)
146
147     # 绘制背景
148     screen.blit(background, (0, 0))
149
150     # 改变飞机图片制造动画
151     if ticks >= ANIMATE_CYCLE:
152         ticks = 0
153     hero.image = hero_surface[ticks//(ANIMATE_CYCLE//2)]
154
155     # 射击
156     if ticks % 10 == 0:
157         hero.single_shoot(bullet1_surface)
158     # 控制子弹
159     hero.bullets1.update()
160     # 绘制子弹
161     hero.bullets1.draw(screen)
162
163     # 产生敌机 *****************************************************
164     if ticks % 30 == 0:
165         enemy = Enemy(enemy1_surface, [randint(0, SCREEN_WIDTH - enemy1_surface.get_width()), -enemy1_surface.get_height()])
166         enemy1_group.add(enemy)
167     # 控制敌机
168     enemy1_group.update()
169     # 绘制敌机
170     enemy1_group.draw(screen)
171     # ************************************************************
172
173     # 检测敌机与子弹的碰撞 *******************************************
174     enemy1_down_group.add(pygame.sprite.groupcollide(enemy1_group, hero.bullets1, True, True))
175
176     for enemy1_down in enemy1_down_group:
177         screen.blit(enemy1_down_surface[enemy1_down.down_index], enemy1_down.rect)
178         if ticks % (ANIMATE_CYCLE//2) == 0:
179             if enemy1_down.down_index < 3:
180                 enemy1_down.down_index += 1
181             else:
182                 enemy1_down_group.remove(enemy1_down)
183     # ************************************************************
184
185     # 绘制飞机
186     screen.blit(hero.image, hero.rect)
187     ticks += 1 # python已略去自增运算符
188
189     # 更新屏幕
190     pygame.display.update()
191
192     # 处理游戏退出
193     # 从消息队列中循环取
194     for event in pygame.event.get():
195         if event.type == pygame.QUIT:
196             pygame.quit()
197             exit()
198
199         # ※ Python中没有switch-case 多用字典类型替代
200         # 控制方向
201         if event.type == pygame.KEYDOWN:
202             if event.key in offset:
203                 offset[event.key] = hero.speed
204         elif event.type == pygame.KEYUP:
205             if event.key in offset:
206                 offset[event.key] = 0
207
208     # 移动飞机
209     hero.move(offset)

时间: 2024-10-27 12:32:44

Pygame - Python游戏编程入门(3)的相关文章

Pygame - Python游戏编程入门(2)

前言 前几天我们做出了一个可控制的飞机,今天我们来做一些小改进,这是代码的一些小改进,却是我们小游戏的一大改进啊~(╯°口°)╯(┴—┴ 然后再引进另外一个主题,pygame.sprite,精灵模块,那它究竟又有什么用呢? 正片开始~ 1. 对主循环的优化 记得我们的上一个版本吗?我们在主循环中不断地绘制背景和飞机,这样的做法其实很消耗cpu资源的,但在这种现象在我们的demo中并不明显,这是为什么呢?我想主要原因应该是我们使用了update()函数(部分刷新,surface与surface之间

Pygame - Python游戏编程入门(4)

前言 一段时间没有敲代码,感觉忘得好快!!今天我们继续完成前面的任务,不知道大家有木有发现之前的飞机撞到敌人是不会爆炸的,这很不符合规律,今天我们加入这个小功能,玩家飞机坠毁并产生动画.(°∀°)? 正片开始~ 1. 判断飞机是否坠毁 关于碰撞检测,我们在上一节的内容中就作了简单介绍了,这一节中我们使用一个新函数,用于判断玩家是否被敌机击中: pygame.sprite.spritecollide()——检测sprite与group之间的碰撞 spritecollide(sprite, grou

Pygame - Python游戏编程入门(1)

前言 在上一篇中,我们初步熟悉了pygame的控制流程,但这对于一个游戏而言是远远不够的.所以在这一篇中,我们的任务是添加一架飞机(玩家),并且能够控制它进行移动,这样我们就又离目标进了一步了~ε=ε=(ノ≧∇≦)ノ 正片开始! 1. 把我们的战斗机搬上屏幕 在正式上代码以前,有一点需要说明的,我们可以看到,所有屏幕上出现的元素都在资源文件(resources/image/shoot.png)中,那我们要怎么做才能把一只飞机给裁剪出来呢?在pygame中,所有在屏幕上显示的元素都可以视为一个su

Pygame - Python游戏编程入门(0)

引言 博客刚开,想把最近学习的东西记录下来,算是一种笔记.最近打算开始学习Python,因为我感觉Python是一门很有意思的语言,很早以前就想学了(碍于懒),它的功能很强大,你可以用它来做科学运算,或者数字图像处理,或者任务的自动化,还可以构建动态网站,很多很多听起来就很有意思的实现.关于Python你还可以找到更多的资料,这里不一一赘述. 一说到开始学习一门新的编程语言,很多童鞋可能马上想到,“嗯,哥去买本书啃啃!”,结果买了本大部头,全是枯燥的语法知识,看了后面忘了前面,事倍功半.记得以前

【python游戏编程之旅】第一篇---初识pygame

本系列博客介绍以python+pygame库进行小游戏的开发.有写的不对之处还望各位海涵. 参考书籍:<python游戏编程入门> 一.pygame简介 Pygame 是一组用来开发游戏软件的 Python 程序模块,基于 SDL 库的基础上开发.允许你在 Python 程序中创建功能丰富的游戏和多媒体程序,Pygame 是一个高可移植性的模块可以支持多个操作系统.用它来开发小游戏非常适合. 可以去http://www.pygame.org/hifi.html 下载并安装使用pygame. 二

游戏编程入门——互动出版网

这篇是计算机类的优质预售推荐>>>><游戏编程入门(第4版)> 游戏编程经典入门读物 内容简介 本书是游戏编程经典入门读物的最新版. 全书共分14章,包含两个附录.本书首先介绍Windows和DirectX编程,然后快速介绍游戏编程的工具箱,包括使用C++和DirectX开发游戏所需的所有基础知识.读者将学习到把思想转化为现实所需的技术,比如2D.3D图形的绘制.背景卷动.处理游戏输入.音效.碰撞检测等.在每章结束时,给出了测验题和项目以便帮助读者实践新学到的技能.本书

【教程】原创:历上最简单的游戏编程入门教程(基于cocos2d-js)

前言: 大家好.我是一个游戏开发者.曾就职于cocos2d-x这个手机游戏引擎的开发的公司. 在这边我准备了一个最简单的教程,想告诉大家制作一个游戏有多简单. 回忆起当年刚刚步入游戏这个行业,我也抱着非常多的疑问. 所以如果大家对游戏有兴趣的朋友,可以在下面留言. 这个教程我会讲的非常通俗易懂.争取几句话之内就让你看到一个效果. 另外教程里面有丰富的图文讲解.我保证你学完之后掌握了做游戏的真髓. 你完全可以马上开始做自己的游戏.并且能够让你的游戏在网页上,ios,android 还有pc平台上跑

Python黑客编程入门教程

我真正开始学习Python是从今年4月份看到2017年的国赛试题之后,之前虽也零星接触过,但都只是皮毛,所以对于我,必须要借助这样一些目标的驱动,才有动力或压力去深入学习下去,这也是我坚持带比赛的一个主要原因. 这套教程是在7月份暑假集训期间录制的,所以从我开始学习Python到录制教程前后也就3个多月的时间,因而教程的内容肯定是相对比较粗糙的.之所以要这么赶,主要是必须要在暑假集训期间让这批准备参加11月份省赛的同学能对Python入门,省赛要跟国赛走,而Python必定是今年省赛的重点.还有

PC游戏编程(入门篇)(前言写的很不错)

PC游戏编程(入门篇) 第一章 基石 1. 1 BOSS登场--GAF简介 第二章 2D图形程式初体验 2.l 饮水思源--第一个"游戏"程式 2.2 知其所以然一一2D图形学基础 2.3 进入图形世界的钥匙--GAFDDraw 2.4 2D图像的本质--图层表面 2.5 场景的秘密--背景卷动 2.6 诱惑--来自"精灵"的问候 2.7 餐后甜点--GAFApp/GAFDDraw的其他法宝 第三章 塞壬的歌声魔力和第三类接触 3.1 1,2,3--计算机音乐概述