一、游戏代码结构

代码结构是代码的组织方式,也是游戏编程的思考框架。pygame官方文档中给出了以下建议,将游戏代码结构分为以下六个部分。

1导入游戏模块。
2资源处理类;定义一些类来处理最基本的资源,包括加载图像和声音,连接和断开网络,加载保存的游戏。
3游戏对象类;为你的游戏对象定义类。例如游戏对象、背景对象、玩家对象、敌人对象、道具对象、文本对象等。
4其他游戏功能;定义其他必要的功能,如游戏面板、菜单处理等。
5初始化游戏,包括pygame对象本身,背景,游戏对象(初始化类的实例)和任何其他你可能想要添加的小代码。
6主循环,你在其中放入任何输入处理(即观察用户敲击按键/鼠标按钮),用于更新游戏对象的代码,最后用于更新屏幕。

基于以上结构,对上一篇最后的代码,我们按照导入游戏模块、定义全局变量、定义游戏对象、定义游戏主循环四个模块了进行优化,其中定义游戏对象又进行了细分。

1.需要导入的游戏模块

import pygame
import sys

2.定义全局常量

分别为screen的size,目标区域的位置,目标区域的size,文本颜色,目标抵达状态。

BG_SIZE = (400, 500)  # 背景尺寸
ON_TARGET = False  # 初始化目标状态
TARGET_SIZE = (50, 50)  # 目标区域大小
TARGET_POSITION = (300, 200)  # 目标位置
TEXT_COLOR = (255, 0, 0)  # 文本颜色

3.定义游戏对象

定义一个Game对象,并定义__init__、event_handler、start三个函数,分别完成所有对象的初始化、事件的监听与更新、游戏主循环。
下面详细介绍:
__init__主要完成游戏的初始化。包括窗口初始化、背景初始化、目标位置初始化、生成玩家实例、生成消息文本实例
代码如下:
>定义Game对象

class Game(object):
    # 游戏类
    def __init__(self):
        # 1.游戏主窗口
        self.screen = pygame.display.set_mode(BG_SIZE)
        self.image = pygame.image.load('./image/plane/background.png')
        self.rect = self.image.get_rect()
        self.stop_position = pygame.image.load('./image/driver/grass.jpg')
        self.stop_position = pygame.transform.scale(self.stop_position, TARGET_SIZE)
        self.stop_position_rect = self.stop_position.get_rect()
        self.stop_position_rect.topleft = TARGET_POSITION
        self.hero = Car()
        self.label = Label()

在Game对象中继续定义event_handler函数。它的任务是使用pygame.event.get()方法监听事件,判断玩家是否点击了窗口的退出按钮,如果是则返回True。

def event_handler(self):
        # 定义事件捕获
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                return True
        return False

在Game对象中定义start函数。它的任务是定义游戏主循环。包括设置循环和屏幕刷新速率、根据用户关闭窗口动作退出主循环、判断是否到达目标、捕获键盘动作、返回移动基数、刷新背景、目标、小车和文本对象、更新屏幕等。
代码如下:

def start(self):
        clock = pygame.time.Clock()
        while True:
            clock.tick(60)
            if self.event_handler():
                return
            if self.hero.rect.topleft == TARGET_POSITION:  # 判断汽车是否移动到目标位置
                ON_TARGET = True
            else:
                ON_TARGET = False
            keys = pygame.key.get_pressed()
            move_hori = keys[pygame.K_RIGHT] - keys[pygame.K_LEFT]  # 获得水平移动基数,向右为正
            move_vert = keys[pygame.K_DOWN] - keys[pygame.K_UP]  # 获得垂直移动基数,向下为正
            self.hero.update(move_hori, move_vert)
            self.label.update(ON_TARGET)
            self.screen.blit(self.image, self.rect)
            self.screen.blit(self.stop_position, self.stop_position_rect)
            self.screen.blit(self.hero.image, self.hero.rect)
            self.screen.blit(self.label.image, self.label.image_rect)
            pygame.display.update()

>定义小车对象
定义了初始化和更新两个函数。
在__init__函数主要用于创建小车surface对象并缩放图片大小,设置小车移动速度。
在update函数主要用于更新小车的坐标位置。具体代码如下:

class Car(object):
    # 定义玩家对象
    def __init__(self):
        self.image = pygame.image.load('./image/driver/car1.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, (50, 50))
        self.rect = self.image.get_rect()
        self.speed = 10

    def update(self, *args):
        self.rect.x += args[0] * self.speed
        self.rect.y += args[1] * self.speed

>定义文本对象
同样定义了初始化和更新两个函数。
在__init__函数主要用于创建生成系统字体对象、文本surface。
在update函数主要用于根据是否到达目标值,刷新文本信息。具体代码如下:

class Label(object):
    # 定义文本对象
    def __init__(self):
        self.font = pygame.font.SysFont('仿宋gb_2312', 25)
        self.image = self.font.render('go to the target', True, TEXT_COLOR)
        self.image_rect = self.image.get_rect()
        self.image_rect.topleft = (100, 100)

    def update(self, is_on_target):
        if is_on_target:
            text = 'well done'
        else:
            text = 'go to the target'
        self.image = self.font.render(text, True, TEXT_COLOR)
        self.image_rect = self.image.get_rect()
        self.image_rect.topleft = (100, 100)

4.定义游戏主程序

def main():
    pygame.init()
    Game().start()
    pygame.quit()


if __name__ == '__main__':
    main()

二、动态效果实现

在游戏中动态效果主要有背景的变化、图像的变化、文本的变化和音乐音效等。以下举例总结一些实现以上动态效果的方法。

1.背景连续滚动

实例:高速飞车

背景连续滚动是通过两个拼接的背景上下移动实现的,如下图所示。

梳理游戏代码架构_python


步骤一:制作Background对象

class Background(object):
    def __init__(self, is_alt):
        self.image = pygame.image.load('./image/driver/road.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, BG_SIZE)
        self.speed = 5  # 设置屏幕移动速度
        self.rect = self.image.get_rect()
        if is_alt:  # 背景初始化时,一个背景在上,一个背景在屏幕内
            self.rect.y = - self.rect.height

    def update(self):
        self.rect.y += self.speed  # 在背景更新时,实现背景向下移动
        if self.rect.y >= self.rect.height: # 当背景刚好移除屏幕时,将背景重新放置到屏幕的上方
            self.rect.y = - self.rect.height

步骤二:在Game类中生成2个背景实例

self.road_1,self.road_2 = Background(True),Background(False)

Game类初始化的完整代码如下:

class Game(object):
    # 游戏类
    def __init__(self):
        # 1.游戏主窗口
        self.screen = pygame.display.set_mode(BG_SIZE)
        self.road_1,self.road_2 = Background(True),Background(False)
        self.speed = 5
        self.hero = Car()  # 生成汽车实例
        self.hero.rect.midbottom = self.screen.get_rect().midbottom  # 在初始化时,将汽车放在屏幕底部中间位置

步骤三:在游戏主循环中添加两个背景更新动作

self.road_1.update()
            self.road_2.update()
            self.screen.blit(self.road_1.image, self.road_1.rect)
            self.screen.blit(self.road_2.image, self.road_2.rect)

完整代码:

import pygame

# 定义全局常量
BG_SIZE = (400, 500)  # 背景尺寸


class Game(object):
    # 游戏类
    def __init__(self):
        # 1.游戏主窗口
        self.screen = pygame.display.set_mode(BG_SIZE)
        self.road_1, self.road_2 = Background(True), Background(False)
        self.speed = 5
        self.hero = Car()
        self.hero.rect.midbottom = self.screen.get_rect().midbottom

    def event_handler(self):
        # 定义事件捕获
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                return True
        return False

    def start(self):
        clock = pygame.time.Clock()
        while True:
            clock.tick(60)
            if self.event_handler():
                return
            keys = pygame.key.get_pressed()
            move_hori = keys[pygame.K_RIGHT] - keys[pygame.K_LEFT]  # 获得水平移动基数,向右为正
            move_vert = keys[pygame.K_DOWN] - keys[pygame.K_UP]  # 获得垂直移动基数,向下为正
            self.road_1.update()
            self.road_2.update()
            self.hero.update(move_hori, move_vert)
            self.screen.blit(self.road_1.image, self.road_1.rect)
            self.screen.blit(self.road_2.image, self.road_2.rect)
            self.screen.blit(self.hero.image, self.hero.rect)
            pygame.display.update()


class Background(object):
    def __init__(self, is_alt):
        self.image = pygame.image.load('./image/driver/road.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, BG_SIZE)
        self.speed = 5
        self.rect = self.image.get_rect()
        if is_alt:
            self.rect.y = - self.rect.height

    def update(self):
        self.rect.y += self.speed
        if self.rect.y >= self.rect.height:
            self.rect.y = - self.rect.height


class Car(object):
    # 定义玩家对象
    def __init__(self):
        self.image = pygame.image.load('./image/driver/car1.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, (50, 50))
        self.rect = self.image.get_rect()
        self.speed = 10

    def update(self, *args):
        self.rect.x += args[0] * self.speed
        self.rect.y += args[1] * self.speed


class Label(object):
    # 定义文本对象
    def __init__(self):
        pass

    def update(self, is_on_target):
        pass


def main():
    pygame.init()
    Game().start()
    pygame.quit()


if __name__ == '__main__':
    main()

2.添加音乐音效

在高速飞车游戏添加四个音乐效果。背景音乐、汽车发动音效、汽车加速行驶音效、汽车喇叭音效。
pygame.mixer.music用以播放背景音乐,pygame.mixer.sound用以播放音效。两者在功能上略有差别。

mixer.Sound

pygame.mixer.Sound 载入音效文件
pygame.mixer.Sound.play 开始声音播放
pygame.mixer.Sound.stop 停止声音播放
pygame.mixer.Sound.fadeout 淡出后停止声音播放
pygame.mixer.Sound.set_volume 设置声音的播放音量
pygame.mixer.Sound.get_volume 获取播放音量
pygame.mixer.Sound.get_num_channels 计算声音播放了多少次
pygame.mixer.Sound.get_length 获取声音的长度
pygame.mixer.Sound.get_raw 返回声音样本的字符串副本

mixer.music

pygame.mixer.music.load 加载音乐文件
pygame.mixer.music.unload 卸载当前加载的音乐以释放资源
pygame.mixer.music.play 开始播放音乐
pygame.mixer.music.rewind 重新开始播放音乐
pygame.mixer.music.stop 停止音乐播放
pygame.mixer.music.pause 暂停音乐播放
pygame.mixer.music.unpause 恢复播放音乐
pygame.mixer.music.fadeout 淡出后停止音乐播放
pygame.mixer.music.set_volume 设置音乐音量
pygame.mixer.music.get_volume 获取音乐音量
pygame.mixer.music.get_busy 检查音乐是否正在播放
pygame.mixer.music.set_pos 设置播放位置
pygame.mixer.music.get_pos 获取音乐播放时间
pygame.mixer.music.queue 将声音文件排队以跟随当前
pygame.mixer.music.set_endevent 让音乐在播放停止時发送一个事件
pygame.mixer.music.get_endevent 获取在播放停止时发送的事件

准备工作
创建一个文件夹,将所有用到的音乐文件放到这个文件夹中。

创建Musicplayer对象
完成音乐音效初始化和设置音乐播放、音效播放、音效停止三个函数。

音效初始化主要是载入音乐、音效文件,为了便于音效文件的文件,可以使用字典进行管理。

代码如下:

class Musicplayer(object):
    def __init__(self):
        pygame.mixer.music.load('./wav/bgm.mp3')  # 载入背景音乐文件
        self.sound_dict = {}  # 生成一个空字典,用于管理音效对象。
        self.sound_dict['car_engine_start'] = pygame.mixer.Sound('./wav/car_engine_start.mp3')  # 载入汽车启动音效
        self.sound_dict['car_speed_up'] = pygame.mixer.Sound('./wav/car_speed_up.wav')  # 载入汽车加速行驶音效
        self.sound_dict['car_beep'] = pygame.mixer.Sound('./wav/car_beep.wav')  # 载入汽车喇叭音效
  • 背景音乐播放函数
def play_music(self):
        pygame.mixer.music.play(-1)  # 当参数为-1时,将循环播放
        pygame.mixer.music.set_volume(0.2)  # 设置播放音量,参数0-1.0之间,1为最大。
  • 音效播放

为了灵活控制不同音效的播放,可将播放的音效名称作为参数传入。

def play_sound(self, wav_name):
        self.sound_dict[wav_name].play()

修改Game对象的start函数设置音乐

游戏开始后播放背景音乐,播放汽车引擎发动音效

def start(self):
        clock = pygame.time.Clock()
        self.music_player.play_music()  # 播放背景音乐
        self.music_player.play_sound('car_engine_start')  # 播放汽车发动音效
        speed = 0  # 设置速度初始值为0,目的是与汽车加速音效相配合

在当汽车任意方向移动基数不为0时,播放汽车加速音效,speed不断加速,最高速度不超过50,将speed作为背景的向下移动速度。

if keys[pygame.K_SPACE]:
                self.music_player.play_sound('car_beep')
            if move_hori != 0 or move_vert != 0:
                self.music_player.play_sound('car_speed_up')
                if speed > 50:
                    speed = 50
                else:
                    speed += 1
            else:
                speed = 0
            self.road_1.update(speed)
            self.road_2.update(speed)

修改Background背景对象代码

class Background(object):
    def __init__(self, is_alt):
        self.image = pygame.image.load('./image/driver/road.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, BG_SIZE)
        self.rect = self.image.get_rect()
        if is_alt:
            self.rect.y = - self.rect.height

    def update(self,speed = 0):
        self.rect.y += speed
        if self.rect.y >= self.rect.height:
            self.rect.y = - self.rect.height

完整代码如下

import pygame
# 定义全局常量
BG_SIZE = (400, 500)  # 背景尺寸


class Game(object):
    # 游戏类
    def __init__(self):
        # 1.游戏主窗口
        self.screen = pygame.display.set_mode(BG_SIZE)
        self.road_1, self.road_2 = Background(True), Background(False)
        self.hero = Car()
        self.music_player = Musicplayer()
        self.hero.rect.midbottom = self.screen.get_rect().midbottom

    def event_handler(self):
        # 定义事件捕获
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                return True
        return False

    def start(self):
        clock = pygame.time.Clock()
        self.music_player.play_music()
        self.music_player.play_sound('car_engine_start')
        speed = 0
        while True:
            clock.tick(60)
            if self.event_handler():
                return
            keys = pygame.key.get_pressed()
            move_hori = keys[pygame.K_RIGHT] - keys[pygame.K_LEFT]  # 获得水平移动基数,向右为正
            move_vert = keys[pygame.K_DOWN] - keys[pygame.K_UP]  # 获得垂直移动基数,向下为正
            if keys[pygame.K_SPACE]:
                self.music_player.play_sound('car_beep')
            if move_hori != 0 or move_vert != 0:
                self.music_player.play_sound('car_speed_up')
                if speed > 50:
                    speed = 50
                else:
                    speed += 1
            else:
                speed = 0
            self.road_1.update(speed)
            self.road_2.update(speed)
            self.hero.update(move_hori, move_vert)
            self.screen.blit(self.road_1.image, self.road_1.rect)
            self.screen.blit(self.road_2.image, self.road_2.rect)
            self.screen.blit(self.hero.image, self.hero.rect)
            pygame.display.update()


class Background(object):
    def __init__(self, is_alt):
        self.image = pygame.image.load('./image/driver/road.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, BG_SIZE)
        self.rect = self.image.get_rect()
        if is_alt:
            self.rect.y = - self.rect.height

    def update(self,speed = 0):
        self.rect.y += speed
        if self.rect.y >= self.rect.height:
            self.rect.y = - self.rect.height


class Musicplayer(object):
    def __init__(self):
        pygame.mixer.music.load('./wav/bgm.mp3')
        self.sound_dict = {}
        self.sound_dict['car_engine_start'] = pygame.mixer.Sound('./wav/car_engine_start.mp3')
        self.sound_dict['car_speed_up'] = pygame.mixer.Sound('./wav/car_speed_up.wav')
        self.sound_dict['car_beep'] = pygame.mixer.Sound('./wav/car_beep.wav')

    def play_music(self):
        pygame.mixer.music.play(-1)
        pygame.mixer.music.set_volume(0.2)

    def play_sound(self, wav_name):
        self.sound_dict[wav_name].play()


class Car(object):
    # 定义玩家对象
    def __init__(self):
        self.image = pygame.image.load('./image/driver/car1.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, (50, 50))
        self.rect = self.image.get_rect()
        self.speed = 10

    def update(self, *args):
        self.rect.x += args[0] * self.speed
        self.rect.y += args[1] * self.speed
        self.rect.x = 0 if self.rect.x <= 0 else self.rect.x
        self.rect.y = 0 if self.rect.y <= 0 else self.rect.y
        if self.rect.right > BG_SIZE[0]:
            self.rect.right = BG_SIZE[0]
        if self.rect.bottom > BG_SIZE[1]:
                self.rect.bottom = BG_SIZE[1]

def main():
    pygame.init()
    Game().start()
    pygame.quit()


if __name__ == '__main__':
    main()

3.动画效果

飞机在飞行时喷出的火焰会发生变化,子弹发射后会飞行一段距离,击中敌人后敌人会受伤甚至被摧毁。以上都属于动画效果。
实现动画效果主要有两种方式,一是图片的变换,二是图片的运动。
实例1:飞机飞行效果
飞机飞行火焰的变化属于图片的轮播效果。实现图片轮播有两种方式,一种是将动画所有帧组成一张长图,按照一定速度同一个方向循环前进,但同一时间只有一帧能被看到,就像坐地铁时看到的广告,视觉上就像是连续播放的视频。第二种将连续动画按关键帧分割成一张张图片,并按找一定频率在图片中依次切换,这和我们平时看动画片是类似的。
以下以第二种方法为例进行介绍。
该方法的操作步骤:
对象初始化
在对象初始化时,生成一张空列表,将图片Surface依次添加到列表中,并设置第一张图片为初始时显示的图片。

class Plane(object):
    # 定义玩家对象
    def __init__(self, rect):
        self.images = []
        self.now_frame = 0  # 初始图片的索引
        self.images.append(pygame.image.load('./image/plane/me1.png').convert_alpha())
        self.images.append(pygame.image.load('./image/plane/me2.png').convert_alpha())
        self.image = self.images[self.now_frame]
        self.rect = self.image.get_rect()
        self.speed = 10

修改游戏主循环代码
对图片切换的间隔进行设置,例如每10帧更换一次。
方法时,在循环中设置当前帧计数器frame_count,初始为0,利用与10取余数的方法,每次循环frame_count会+1,到第10次时归为0。最后将frame_count==0返回给对象的update函数。

def start(self):
        clock = pygame.time.Clock()
        frame_count = 0
        while True:
            clock.tick(40)
            if self.event_handler():
                return
            frame_count = (frame_count + 1) % 10
            self.hero.update(frame_count==0, move_hori, move_vert)

设置对象更新函数
当对象update函数被调用时,先判断args[0]是否为真,即判断frame_count == 0 是否为真,如果为True则代表此次时过去了10帧,需切换图片;如果为false,则直接返回。

更新的方式是从头开始轮播图片,如果当前不是最后一张图片,则切换到下一张;如果是最后一张图片,则切换到第一张。

def update(self, *args):
        if not args[0]:
            return
        if self.now_frame < range(len(self.images))[-1]:
            self.now_frame += 1
            self.image = self.images[self.now_frame]
        else:
            self.now_frame = range(len(self.images))[0]
            self.image = self.images[self.now_frame]

本篇主要内容如上,主要介绍了游戏代码的典型结构,总结了背景连续滚动、声音效果和图像动态效果的实现方法。下一篇将对Pygame的精灵和精灵组实现方法进行。

上例完整代码如下:

import pygame
import time
import os
# 定义全局常量
BG_SIZE = (400, 500)  # 背景尺寸
INIT_RECT = pygame.Rect(200, 250, 60, 60)

class Game(object):
    # 游戏类
    def __init__(self):
        # 1.游戏主窗口
        self.screen = pygame.display.set_mode(BG_SIZE)
        self.road_1, self.road_2 = Background(True), Background(False)
        self.hero = Plane(INIT_RECT)
        self.hero.rect.midbottom = self.screen.get_rect().midbottom

    def event_handler(self):
        # 定义事件捕获
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                return True
        return False

    def start(self):
        clock = pygame.time.Clock()
        frame_count = 0
        while True:
            clock.tick(40)
            if self.event_handler():
                return
            keys = pygame.key.get_pressed()
            move_hori = keys[pygame.K_RIGHT] - keys[pygame.K_LEFT]  # 获得水平移动基数,向右为正
            move_vert = keys[pygame.K_DOWN] - keys[pygame.K_UP]  # 获得垂直移动基数,向下为正
            self.road_1.update()
            self.road_2.update()
            frame_count = (frame_count + 1) % 10
            self.hero.update(frame_count == 0, move_hori, move_vert)
            self.screen.blit(self.road_1.image, self.road_1.rect)
            self.screen.blit(self.road_2.image, self.road_2.rect)
            self.screen.blit(self.hero.image, self.hero.rect)
            pygame.display.update()


class Background(object):
    def __init__(self, is_alt):
        self.image = pygame.image.load('./image/plane/background.png').convert_alpha()
        self.image = pygame.transform.scale(self.image, BG_SIZE)
        self.rect = self.image.get_rect()
        if is_alt:
            self.rect.y = - self.rect.height

    def update(self,speed = 1):
        self.rect.y += speed
        if self.rect.y >= self.rect.height:
            self.rect.y = - self.rect.height



class Plane(object):
    # 定义玩家对象
    def __init__(self, rect):
        self.images = []
        self.now_frame = 0
        self.images.append(pygame.image.load('./image/plane/me1.png').convert_alpha())
        self.images.append(pygame.image.load('./image/plane/me2.png').convert_alpha())
        self.image = self.images[self.now_frame]
        self.rect = self.image.get_rect()
        self.speed = 10

    def update(self, *args):
        if not args[0]:
            return
        if self.now_frame < range(len(self.images))[-1]:
            self.now_frame += 1
            print(self.now_frame)
            self.image = self.images[self.now_frame]
        else:
            self.now_frame = range(len(self.images))[0]
            print(self.now_frame)
            self.image = self.images[self.now_frame]

        self.rect.x += args[1] * self.speed
        self.rect.y += args[2] * self.speed
        self.rect.x = 0 if self.rect.x <= 0 else self.rect.x
        self.rect.y = 0 if self.rect.y <= 0 else self.rect.y
        if self.rect.right > BG_SIZE[0]:
            self.rect.right = BG_SIZE[0]
        if self.rect.bottom > BG_SIZE[1]:
                self.rect.bottom = BG_SIZE[1]


class Label(object):
    # 定义文本对象
    def __init__(self):
        pass

    def update(self, is_on_target):
        pass


def main():
    pygame.init()
    Game().start()
    pygame.quit()