注册 登录
编程论坛 Python论坛

用python怎么做贪吃蛇

一张纸白 发布于 2023-09-01 10:09, 883 次点击
用python怎么做贪吃蛇
9 回复
#2
一张白纸1232023-09-02 11:21
建议你要先拥有一个第三方库:Pygame
#3
一张白纸1232023-09-02 11:28
程序代码如下:
import pygame,sys
from pygame.locals import *
from sys import exit

class Snake(pygame.sprite.Sprite):

    snake_head=pygame.image.load('snake_head.png')
    snake_body=pygame.image.load('snake_body.png')
    snake_tail=pygame.image.load('snake_tail.png')
   
    def __init__(self,screen):
        pygame.sprite.Sprite.__init__(self)
        #蛇序列
        self.snakeLine=[(400,250),(400-40,250),(400-40*2,250)]

        #画板
        self.screen=screen

        #碰撞检测
        self.rect=Rect(400,250,40,40) #蛇头的区域

    def show(self,newSnakeLine=None):
        #将新的蛇序列设为原蛇序列
        if newSnakeLine is not None:
            self.snakeLine=newSnakeLine
            self.rect=Rect(newSnakeLine[0],(40,40))

        self.screen.blit(Snake.snake_head,self.snakeLine[0])
        self.rect=Rect(self.snakeLine[0],(40,40))

        #蛇尾
        for cord in self.snakeLine[1:-1]:
            self.screen.blit(Snake.snake_body,cord)

        #蛇尾
        self.screen.blit(Snake.snake_tail,self.snakeLine[-1])
        
    def getDirection(self):
        offX=self.snakeLine[0][0]-self.snakeLine[1][0]
        offY=self.snakeLine[0][1]-self.snakeLine[1][1]
        if (offX==30 or offX==-870) and offY==0:
            return K_RIGHT
        
        elif (offX==-30 or offX==870) and offY==0:
            return K_LEFT
        
        elif offX==0 and (offY==30 or offY==-570):
            return K_DOWN
        
        elif offX==0 and (offY==-30 or offY==570):
            return K_UP

    def move(self,KEY=None):
   
        #创建一个冲突列表
        wrong_list=[{K_RIGHT,K_RIGHT},{K_RIGHT,K_LEFT},{K_LEFT,K_LEFT},{K_UP,K_UP},{K_UP,K_DOWN},{K_DOWN,K_DOWN}]
        if KEY is None or {self.getDirection(),KEY} in wrong_list:
            KEY=self.getDirection()#按错方向键后,不改变原方向
            
        if KEY==K_RIGHT:#右移就在右侧添加蛇头,蛇尾减少1块
            self.snakeLine=[((self.snakeLine[0][0]+40)%900,self.snakeLine[0][1])]+self.snakeLine[0:-1]
            
        elif KEY==K_UP:#上移就在上侧添加蛇头,蛇尾减少1块
            self.snakeLine=[(self.snakeLine[0][0],(600+self.snakeLine[0][1]-40)%600)]+self.snakeLine[0:-1]
            
        elif KEY==K_DOWN:#下移就在下侧添加蛇头,蛇尾减少1块
            self.snakeLine=[(self.snakeLine[0][0],(self.snakeLine[0][1]+40)%600)]+self.snakeLine[0:-1]
            
        elif KEY==K_LEFT:#左移就在左侧添加蛇头,蛇尾减少1块
            self.snakeLine=[((self.snakeLine[0][0]-40+900)%900,self.snakeLine[0][1])]+self.snakeLine[0:-1]
            
        self.show()

pygame.init()

screen=pygame.display.set_mode((900,600))
pygame.display.set_caption('snake')

snake=Snake(screen)

clock=pygame.time.Clock()


while True:
   
    clock.tick(10)

    KEY=None
   
    for event in pygame.event.get():#进行事件等待
        #如果用户单击关闭窗口按钮就退出
        if event.type==QUIT:
            pygame.quit()
            exit()
        if event.type==KEYDOWN:
            KEY=event.key

    screen.fill('black')

    snake.move(KEY)

    #循环显示
    pygame.display.update()
#4
一张白纸1232023-09-02 11:29
但其中的素材你要自己找
也可以用矩形rect代替
#5
小白在此2023-09-02 13:47
程序代码:
import random,sys,pygame
from pygame.locals import*
FPS=5
width=640
heigh=480
cellsize=20
cellwidth=int(width/cellsize)
cellheigh=int(heigh/cellsize)
white=(255,255,255)
black=(0,0,0)
red=(255,0,0)
green=(0,255,0)
blue=(0,0,255)
darkgreen=(0,155,0)
darkgray=(40,40,40)
BGcolor=white
Up='up'
Down='down'
Left='left'
Right='right'
def main():
    global FPSlock,screen,BASICfont
    pygame.init()
    FPSlock=pygame.time.Clock()
    screen=pygame.display.set_mode([width,heigh])
    BASICfont=pygame.font.Font(r'C:\Windows\Fonts\simkai.ttf',20)
    pygame.display.set_caption('贪吃蛇')
    showstartscreen()
    while True:
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                sys.exit()
        rungame()
        showgameover()
def showstartscreen():
    width1=640
    heigh1=480
    titlefont=pygame.font.Font(r'C:\Windows\Fonts\simkai.ttf',100)
    titlesurf1=titlefont.render('贪吃蛇',True,black,darkgreen)
    titlesurf2=titlefont.render('贪吃蛇',True,green)
    degrees1=0
    degrees2=0
    while True:
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                sys.exit()
            elif event.type == KEYDOWN:
                return
        screen.fill(BGcolor)
        surf1=pygame.transform.rotate(titlesurf1,degrees1)
        rect1=surf1.get_rect()
        rect1.center=(width1/2,heigh1/2)
        screen.blit(surf1,rect1)
        surf2=pygame.transform.rotate(titlesurf2,degrees2)
        rect2=surf2.get_rect()
        rect2.center=(width1/2,heigh1/2)
        screen.blit(surf2,rect2)
        drawpresskeymsg()
        pygame.display.update()
        FPSlock.tick(FPS)
        degrees1 += 3
        degrees2 += 7
def drawpresskeymsg():
    presskeysurf=BASICfont.render('按任意键进入游戏',True,black)
    presskeyrect=presskeysurf.get_rect()
    presskeyrect.topleft=(width-200,heigh-30)
    screen.blit(presskeysurf,presskeyrect)
def drawgrid():
    for x in range(0,width,cellsize):
        pygame.draw.line(screen,darkgreen,(x,0),(x,heigh))
    for y in range(0,heigh,cellsize):
        pygame.draw.line(screen,darkgreen,(0,y),(width,y))
def getrandomlocation():
    return {'x':random.randint(0,cellwidth),'y':random.randint(0,cellheigh-1)}
def drawscore(score):
    scoresurf=BASICfont.render('分数:%s'%(score),True,black)
    scorerect=scoresurf.get_rect()
    scorerect.topleft=(width-120,10)
    screen.blit(scoresurf,scorerect)
def rungame():
    startx=random.randint(5,cellwidth-6)
    starty=random.randint(5,cellheigh-6)
    wormcoords=[{'x':startx,'y':starty},{'x':startx-1,'y':starty},{'x':startx-2,'y':starty}]
    direction=Right
    apple=getrandomlocation()
    while True:
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                sys.exit()
            if event.type == KEYDOWN:
                if (event.key==K_LEFT or event.key==K_a)and direction != Right:
                    direction=Left
                elif (event.key==K_RIGHT or event.key==K_d)and direction != Left:
                    direction=Right
                elif (event.key==K_UP or event.key==K_w)and direction != Down:
                    direction=Up
                elif (event.key==K_DOWN or event.key==K_s)and direction != Up:
                    direction=Down
                elif event.key == K_ESCAPE:
                    sys.exit()
        if wormcoords[0]['x'] == -1 or wormcoords[0]['x'] == cellwidth or wormcoords[0]['y'] == -1 or wormcoords[0]['y'] == cellheigh:
            return
        for wormbody in wormcoords[1:]:
            if wormbody['x'] == wormcoords[0]['x'] and wormbody['y'] == wormcoords[0]['y']:
                return
        if wormcoords[0]['x'] == apple['x'] and wormcoords[0]['y'] == apple['y']:
            apple= getrandomlocation()
        else:
            del wormcoords[-1]
        if direction == Up:
            newhead={'x':wormcoords[0]['x'],'y':wormcoords[0]['y']-1}
        elif direction == Down:
            newhead={'x':wormcoords[0]['x'],'y':wormcoords[0]['y']+1}
        elif direction == Left:
            newhead={'x':wormcoords[0]['x']-1,'y':wormcoords[0]['y']}
        elif direction == Right:
            newhead={'x':wormcoords[0]['x']+1,'y':wormcoords[0]['y']}
        wormcoords.insert(0,newhead)
        screen.fill(BGcolor)
        drawgrid()
        drawworm(wormcoords)
        drawapple(apple)
        drawscore(len(wormcoords)-3)
        pygame.display.update()
        FPSlock.tick(FPS)
def drawworm(wormcoords):
    for coord in wormcoords:
        x=coord['x']*cellsize
        y=coord['y']*cellsize
        wormsegmentrect=pygame.Rect(x,y,cellsize,cellsize)
        pygame.draw.rect(screen,darkgreen,wormsegmentrect)
        worminnersegmentrect=pygame.Rect(x+4,y+4,cellsize-8,cellsize-8)
        pygame.draw.rect(screen,darkgreen,worminnersegmentrect)
def drawapple(coord):
    x=coord['x']*cellsize
    y=coord['y']*cellsize
    applerect=pygame.Rect(x,y,cellsize,cellsize)
    pygame.draw.rect(screen,red,applerect)
def showgameover():
    gameoverfont=pygame.font.Font('C://Windows//Fonts//msyh.ttc',150)
    gamesurf=gameoverfont.render('游戏',True,black)
    oversurf=gameoverfont.render('失败',True,black)
    gamerect=gamesurf.get_rect()
    overrect=oversurf.get_rect()
    gamerect.midtop=(width/2,10)
    overrect.midtop=(width/2,gamerect.height+10+25)
    screen.blit(gamesurf,gamerect)
    screen.blit(oversurf,overrect)
    drawpresskeymsg()
    pygame.display.update()
    while True:
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                sys.exit()
            return
if __name__=='__main__':
    main()

亲测可用
#6
一张纸白2023-09-02 15:44
谢谢
#7
一张纸白2023-09-02 15:45
不过我这没有分
#8
一张白纸1232023-09-02 15:48
厉害呀,大神
#9
小白在此2023-09-02 17:20
回复 7楼 一张纸白
不会啊,分数在右上角啊。
如果你觉得满意的话,能不能结个帖呢

[此贴子已经被作者于2023-9-9 11:21编辑过]

#10
沉着的导弹鲨2024-11-28 19:30
程序代码:
#!/usr/bin/env python
#
-*- coding: utf-8 -*-
#
author:Wangdali time:2021年1月20日16:08:44
#
python实现:贪吃蛇
"""
游戏玩法:回车开始游戏;空格暂停游戏/继续游戏;方向键/wsad控制小蛇走向
"""
'''
思路:用列表存储蛇的身体;用浅色表示身体,深色背景将身体凸显出来;
蛇的移动:仔细观察,是:身体除头和尾不动、尾部消失,头部增加,所以,新添加的元素放在列表头部、删除尾部元素;
游戏结束判定策略:超出边界;触碰到自己的身体:蛇前进的下一格子为身体的一部分(即在列表中)。
'''
# 注:因为在列表中需要频繁添加和删除元素,所以用deque容器代替列表;是因为deque具有高效的插入和删除效率
#
初始化蛇,长度为3,放置在屏幕左上角;
#
导包
import random
import sys
import time
import pygame
from pygame.locals import *
from collections import deque

# 基础设置
Screen_Height = 480
Screen_Width = 600
Size = 20  # 小方格大小
Line_Width = 1
# 游戏区域的坐标范围
Area_x = (0, Screen_Width // Size - 1)  # 0是左边界,1是右边界 #注:python中//为整数除法;/为浮点数除法
Area_y = (2, Screen_Height // Size - 1)
# 食物的初步设置
#
食物的分值+颜色
Food_Style_List = [(10, (255, 100, 100)), (20, (100, 255, 100)), (30, (100, 100, 255))]
# 整体颜色设置
Light = (100, 100, 100)
Dark = (200, 200, 200)
Black = (0, 0, 0)
Red = (200, 30, 30)
Back_Ground = (40, 40, 60)


# 文本输出格式设置
def Print_Txt(screen, font, x, y, text, fcolor=(255, 255, 255)):
    # font.render参数意义:.render(内容,是否抗锯齿,字体颜色,字体背景颜色)
    Text = font.render(text, True, fcolor)
    screen.blit(Text, (x, y))


# 初始化蛇
def init_snake():
    snake = deque()
    snake.append((2, Area_y[0]))
    snake.append((1, Area_y[0]))
    snake.append((0, Area_y[0]))
    return snake


# 食物设置
#
注意需要对食物出现在蛇身上的情况进行判断
def Creat_Food(snake):
    """
    注:randint 产生的随机数区间是包含左右极限的,
    也就是说左右都是闭区间的[1, n],能取到1和n。
    而 randrange 产生的随机数区间只包含左极限,
    也就是左闭右开的[1, n),1能取到,而n取不到。randint
    产生的随机数是在指定的某个区间内的一个值,
    而 randrange 产生的随机数可以设定一个步长,也就是一个间隔。
   
"""
    food_x = random.randint(Area_x[0], Area_x[1])  # 此处有疑问
    food_y = random.randint(Area_y[0], Area_y[1])
    # 如果食物出现在蛇上,重来;
    while (food_x, food_y) in snake:
        food_x = random.randint(Area_x[0], Area_x[1])
        food_y = random.randint(Area_y[0], Area_y[1])
    return food_x, food_y


# 食物风格
def Food_Style():
    return Food_Style_List[random.randint(0, 2)]  # 返回随机的分值和颜色


def main():
    pygame.init()
    screen = pygame.display.set_mode((Screen_Width, Screen_Height))  # 初始化一个准备显示的窗口或屏幕
    pygame.display.set_caption('贪吃蛇')  # Set the current window caption
    # 得分字体设置
    font1 = pygame.font.SysFont('SimHei', 24)
    # GO字体设置
    font2 = pygame.font.SysFont(None, 72)
    fwidth, fheight = font2.size('GAME OVER')  ###
    # 程序bug修复:如果蛇在向右移动,快速点击分别施加向下、向左的命令,向下的命令会被覆盖,只有向左的命令被接受,直接GameOver
    # b变量为了防止这个情况发生
    b = True
    # 蛇z
    snake = init_snake()
    # 食物
    food = Creat_Food(snake)
    food_style = Food_Style()
    # 方向控制
    pos = (1, 0)  ###
    # 启动游戏相关变量初始化
    game_over = True  # 结束标志 # 是否开始,当start = True,game_over = True 时,才显示 GAME OVER
    game_start = False  # 开始标志
    score = 0  # 得分
    orispeed = 0.3  # 蛇初始速度
    speed = orispeed  # 蛇速度
    last_move_time = None
    pause = False  # 暂停
    while True:
        for event in pygame.event.get():
            if event.type == QUIT:
                sys.exit()
            elif event.type == KEYDOWN:
                if event.key == K_RETURN:
                    if game_over:
                        game_start = True
                        game_over = False
                        b = True
                        snake = init_snake()
                        food = Creat_Food(snake)
                        food_style = Food_Style()
                        pos = (1, 0)
                        # 得分
                        score = 0
                        last_move_time = time.time()
                elif event.key == K_SPACE:
                    if not game_over:
                        pause = not pause
                # 以下为防止蛇在向右移动时按向左键,导致GameOver
                elif event.key in (K_UP, K_w):
                    if b and not pos[1]:  ###
                        pos = (0, -1)
                        b = False
                elif event.key in (K_DOWN, K_s):
                    if b and not pos[1]:
                        pos = (0, 1)
                        b = False
                elif event.key in (K_LEFT, K_a):
                    if b and not pos[0]:
                        pos = (-1, 0)
                        b = False
                elif event.key in (K_RIGHT, K_d):
                    if b and not pos[0]:
                        pos = (1, 0)
                        b = False
        # 填充背景色
        screen.fill(Back_Ground)
        ###
        # 画网格线、竖线
        for x in range(Size, Screen_Width, Size):
            pygame.draw.line(screen, Black, (x, Area_y[0] * Size), (x, Screen_Height), Line_Width)
        # 画网格线、横线
        for y in range(Area_y[0] * Size, Screen_Height, Size):
            pygame.draw.line(screen, Black, (0, y), (Screen_Width, y), Line_Width)
        # 蛇的爬行过程
        if not game_over:
            curTime = time.time()
            if curTime - last_move_time > speed:  ###
                if not pause:
                    b = True
                    last_move_time = curTime
                    next_s = (snake[0][0] + pos[0], snake[0][1] + pos[1])
                    # 如果吃到了食物
                    if next_s == food:
                        snake.appendleft(next_s)
                        score += food_style[0]
                        speed = orispeed - 0.03 * (score // 100)
                        food = Creat_Food(snake)
                        food_style = Food_Style()
                    else:
                        # 在区域内
                        if Area_x[0] <= next_s[0] <= Area_x[1] and Area_y[0] <= next_s[1] <= Area_y[1] and next_s not in snake:
                            snake.appendleft(next_s)
                            snake.pop()
                        else:
                            game_over = True
        # 画食物
        if not game_over:
            '''
        rect(Surface,color,Rect,width=0)
第一个参数指定矩形绘制到哪个Surface对象上

第二个参数指定颜色

第三个参数指定矩形的范围(left,top,width,height)

第四个参数指定矩形边框的大小(0表示填充矩形)

例如绘制三个矩形:

    pygame.draw.rect(screen, BLACK, (50, 50, 150, 50), 0)
    pygame.draw.rect(screen, BLACK, (250, 50, 150, 50), 1)
    pygame.draw.rect(screen, BLACK, (450, 50, 150, 50), 10)
         
'''
        # 避免 GAME OVER 的时候把 GAME OVER 的字给遮住了
        pygame.draw.rect(screen, food_style[1], (food[0] * Size, food[1] * Size, Size, Size), 0)
        # 画蛇
        for s in snake:
            pygame.draw.rect(screen, Dark, (s[0] * Size + Line_Width, s[1] * Size + Line_Width,
                                            Size - Line_Width * 2, Size - Line_Width * 2), 0)
        Print_Txt(screen, font1, 30, 7, f'速度: {score // 100}')
        Print_Txt(screen, font1, 450, 7, f'得分: {score}')
        # 画GameOver
        if game_over:

            if game_start:
                # print('GameOver')
                Print_Txt(screen, font2, (Screen_Width - fwidth) // 2, (Screen_Height - fheight) // 2, 'GAME OVER', Red)
        pygame.display.update()


if __name__ == '__main__':
    main()

这个也可以实现
1