pygame制作贪吃蛇游戏教程(二)

2020-03-14  本文已影响0人  曾大播

接下来先让蛇动起来,有两种思路:
1、全局更新:隔一段时间全部重画。这样写比较简单,但是比较消耗内存,可能运行不顺畅
2、局部更新:擦掉头和尾巴的一小部分重画
考虑到后期蛇比较长的时候,如果每次运行都重新画,画面肯定是不流畅的,因此这里采用局部更新的方法

准备工作

    while True:
        for event in pygame.event.get():

这条语句实际上是以相当快的频率进行消息循环的。我们的游戏不需要这么高帧,因此需要自己定义一个事件,隔一段时间运行一次,在main.py中定义全局变量,要用的时候再settimer:

#定义
SNAKE_MOVE = pygame.USEREVENT
#调用
    pygame.time.set_timer(SNAKE_MOVE, 1000)
#取消调用
    pygame.time.set_timer(SNAKE_MOVE, 0)

这样就每隔1s调用一次SNAKE_MOVE事件了
为了调试的时候舒服,先让它能够随时暂停和继续:

# is_pause是新定义的变量,按p键暂停或开始
            if event.type == KEYDOWN:
                if event.key == pygame.K_p:
                    if is_pause == 0:
                        is_pause = 1
                        pygame.time.set_timer(SNAKE_MOVE, 0)
                    else:
                        is_pause = 0
                        pygame.time.set_timer(SNAKE_MOVE, 1000)

正式开始

下面开始编写SNAKE_MOVE函数,思路是只动头尾。
由于涉及擦除图像的一部分,因此先定一下背景,这里设为原谅绿。
背景插入图像步骤是一样的,也是screen.blit,不再赘述,最后会有完整代码。
具体思路:
1、找到头尾
2、把头和尾处擦掉,画上新头和旧头,这样每次只需要更新四个图,有效减少了内存的浪费
上代码(昨天写的逻辑上有修改,不过很小):
main.py:

import pygame
from pygame import *
from settings import Settings
from snake import Snake
import sys

SNAKE_MOVE = pygame.USEREVENT

def main():
    is_pause = 0
    pygame.init()  #初始化背景设置
    settings = Settings()  # 实例化
    snake = Snake(settings)
    screen = pygame.display.set_mode((settings.screen_width, settings.screen_height)) #调用属性设置屏幕的宽高
    pygame.display.set_caption("贪吃蛇")  #设置标题
    background = pygame.image.load("../Assets/background.jpg")
    screen.blit(background, (0, 0))
    snake.draw_snake(screen)
    pygame.time.set_timer(SNAKE_MOVE, 300)
    while True:
        for event in pygame.event.get():
            if event.type == pygame.QUIT:#点击右上角的红叉关闭程序
                sys.exit()
            if event.type == SNAKE_MOVE:
                snake.redraw_snake(screen)
            if event.type == KEYDOWN:
                if event.key == pygame.K_p:
                    if is_pause == 0:
                        is_pause = 1
                        pygame.time.set_timer(SNAKE_MOVE, 0)
                    else:
                        is_pause = 0
                        pygame.time.set_timer(SNAKE_MOVE, 300)
main()

snake.py:

#snake.py
#基础设置参数都在这里
from numpy import *
import pygame
from pygame import *
from snakepic import SnakePic
class Snake():
    def __init__(self, settings):
        self.current_direction = 2  #1表示向上,2右,3下,4左
        self.settings = settings
        # 根据这个跟地图同维矩阵来确定蛇长什么样子
        # 0表示空,1、2、3、4表示方向,蛇在地图这个点的方向
        self.discribe = zeros((settings.map_x+2, settings.map_y+2), dtype=int)
        # 更详细的蛇的样子,0表示空,1~4表示头,5~8表示拐弯,9~12表示身子,13~16表示尾巴
        self.detail_discribe = zeros((settings.map_x+2, settings.map_y+2), dtype=int)
        self.read_snake_data()

    def detail_snake(self):
        for i in range(0, self.settings.map_x+2):
            for k in range(0, self.settings.map_y+2):
                if self.discribe[i, k] == 0:
                    self.detail_discribe[i, k] = 0
                    continue
                self.detail_discribe[i, k] = self.detail_judge(i, k)

    def detail_judge(self, i, k):
        if i == 1 and k == 2:
            c=0
        if self.discribe[i, k] == 1:
            if self.discribe[i+1, k] == 1 and self.discribe[i-1, k] != 0:
                return 9
            if self.discribe[i-1, k] == 0:
                return 1
            if self.discribe[i+1, k] != 1:
                if self.discribe[i, k-1] != 2 and self.discribe[i, k+1] != 4:
                    return 13
                if self.discribe[i, k+1] == 4:
                    return 7
                if self.discribe[i, k-1] == 2:
                    return 6

        if self.discribe[i, k] == 2:
            if self.discribe[i, k+1] != 0 and self.discribe[i, k-1] == 2:
                return 10
            if self.discribe[i, k+1] == 0:
                return 2
            if self.discribe[i, k-1] != 2:
                if self.discribe[i+1, k] != 1 and self.discribe[i-1, k] != 3:
                    return 14
                if self.discribe[i+1, k] == 1:
                    return 8
                if self.discribe[i-1, k] == 3:
                    return 7

        if self.discribe[i, k] == 3:
            if self.discribe[i+1, k] != 0 and self.discribe[i-1, k] == 3:
                return 11
            if self.discribe[i+1, k] == 0:
                return 3
            if self.discribe[i-1, k] != 3:
                if self.discribe[i, k+1] != 4 and self.discribe[i, k-1] != 2:
                    return 15
                if self.discribe[i, k+1] == 4:
                    return 8
                if self.discribe[i, k-1] == 2:
                    return 5

        if self.discribe[i, k] == 4:
            if self.discribe[i, k+1] == 4 and self.discribe[i, k-1] != 0:
                return 12
            if self.discribe[i, k-1] == 0:
                return 4
            if self.discribe[i, k+1] != 4:
                if self.discribe[i+1, k] != 1 and self.discribe[i-1, k] != 3:
                    return 16
                if self.discribe[i+1, k] == 1:
                    return 5
                if self.discribe[i-1, k] == 3:
                    return 6
        c=0

    def draw_snake(self, screen):
        self.detail_snake()
        for i in range(0, self.settings.map_x+2):
            for k in range(0, self.settings.map_y+2):
                if self.detail_discribe[i, k] == 0:
                    continue
                temp = SnakePic(self.detail_discribe[i, k])
                temp.blitme(screen, i, k)
        pygame.display.update()

    def read_snake_data(self):
        f = open('../Data/level1.txt')
        lines = f.readlines()
        A_row = 1
        for line in lines:
            list = line.split('\t')
            self.discribe[A_row, 1: self.settings.map_x + 1] = list[0: self.settings.map_x]
            A_row += 1

    def redraw_snake(self, screen):
        self.detail_snake()
        #先找到头和尾
        head = [0, 0]
        tail = [0, 0]
        head_new = [0, 0]
        tail_new = [0, 0]
        for i in range(0, self.settings.map_x+2):
            for k in range(0, self.settings.map_y+2):
                if self.detail_discribe[i, k] == 1:
                    head = [i, k]
                    head_new = [i-1, k]
                if self.detail_discribe[i, k] == 2:
                    head = [i, k]
                    head_new = [i, k+1]
                if self.detail_discribe[i, k] == 3:
                    head = [i, k]
                    head_new = [i+1, k]
                if self.detail_discribe[i, k] == 4:
                    head = [i, k]
                    head_new = [i, k-1]
                if self.detail_discribe[i, k] == 13:
                    tail = [i, k]
                    tail_new = [i-1, k]
                if self.detail_discribe[i, k] == 14:
                    tail = [i, k]
                    tail_new = [i, k+1]
                if self.detail_discribe[i, k] == 15:
                    tail = [i, k]
                    tail_new = [i+1, k]
                if self.detail_discribe[i, k] == 16:
                    tail = [i, k]
                    tail_new = [i, k-1]
        P = [head, head_new, tail_new]
        if self.discribe[head[0], head[1]] == 1:
            self.discribe[head[0]-1, head[1]] = 2
        if self.discribe[head[0], head[1]] == 2:
            self.discribe[head[0], head[1]+1] = 2
        if self.discribe[head[0], head[1]] == 3:
            self.discribe[head[0]+1, head[1]] = 2
        if self.discribe[head[0], head[1]] == 2:
            self.discribe[head[0], head[1]-1] = 2
        self.discribe[tail[0], tail[1]] = 0
        rect1 = pygame.Rect(50*(head[1]-1), 50*(head[0]-1), 50, 50)
        rect2 = pygame.Rect(50*(tail[1]-1), 50*(tail[0]-1), 50, 50)
        rect3 = pygame.Rect(50*(head_new[1]-1), 50*(head_new[0]-1), 50, 50)
        rect4 = pygame.Rect(50*(tail_new[1]-1), 50*(tail_new[0]-1), 50, 50)
        #把新头、旧头、新尾、旧尾擦掉
        screen.fill((0, 255, 0), rect1)
        screen.fill((0, 255, 0), rect2)
        screen.fill((0, 255, 0), rect3)
        screen.fill((0, 255, 0), rect4)
        #看看新蛇啥样
        self.detail_snake()
        for p in P:
            if self.detail_discribe[p[0], p[1]] == 0:
                c=0
            temp = SnakePic(self.detail_discribe[p[0], p[1]])
            temp.blitme(screen, p[0], p[1])
        pygame.display.update()

效果如下:


贪吃蛇 2020-03-14 12-58-57.gif

可见,画出来有点断断续续的,接下来会进行改进,让其变的流畅。
(未完待续)

上一篇下一篇

猜你喜欢

热点阅读