贪吃蛇

初始版本

初始版本,只存在基本数据结构——双向队列。

游戏思路

贪吃蛇通过不断得吃食物来增长自身,如果贪吃蛇碰到边界或者自身则游戏失败。

食物是绿色矩形来模拟,坐标为随机数生成,定义一个蛇长变量,判断蛇头坐标和食物坐标是否接近,如果蛇头接近食物,蛇长增加一个单位。

蛇结构体通过双向队列实现蛇的移动和增长。用pygame相应库函数读取键盘事件,每次事件发生都对应蛇头的相应方向,例如按键盘下键贪吃蛇向下走。蛇的移动是通过在队头添加一个新的坐标,然后删掉队尾元素实现的。队列的长度始终为蛇长变量的值。

得分变量为蛇长变量减去一个单位,初始化的时候蛇长就为1个单位长度,故而需要减1。

完整代码

import pygame
import random pygame.init() # 以RGB的形式定义颜色
white = (255, 255, 255)
yellow = (255, 255, 102)
black = (0, 0, 0)
red = (213, 50, 80)
green = (0, 255, 0)
blue = (50, 153, 213) # 设置窗体大小
dis_width = 800
dis_height = 600
dis = pygame.display.set_mode((dis_width, dis_height))
pygame.display.set_caption('贪吃蛇 KevenDuan1.0') # 创建clock对象
clock = pygame.time.Clock() # 蛇的宽度
snake_block = 15
# 蛇的速度
snake_speed = 15 # 从系统库里获取字体
font_style = pygame.font.SysFont("bahnschrift", 25)
score_font = pygame.font.SysFont("comicsansms", 35) def Your_score(score):
value = score_font.render("Score: " + str(score), True, yellow)
# 在主surface里添加字体surface
dis.blit(value, [0, 0]) def our_snake(snake_block, snake_list):
for x in snake_list:
pygame.draw.rect(dis, black, [x[0], x[1], snake_block, snake_block]) def message(msg, color):
mesg = font_style.render(msg, True, color)
dis.blit(mesg, [dis_width // 2 - 180, dis_height // 2]) def gameLoop():
game_over = False
game_close = False x1 = dis_width / 2
y1 = dis_height / 2 x1_change = 0
y1_change = 0 # 存放蛇的身体
snake_List = []
# 蛇的长度
Length_of_snake = 1 # 食物坐标随机生成
foodx = round(random.randrange(snake_block, dis_width - snake_block, snake_block))
foody = round(random.randrange(snake_block, dis_height - snake_block, snake_block))
print(foodx, foody) while not game_over:
while game_close == True:
dis.fill(blue)
message("Game over, press p again or q quit!", red)
Your_score(Length_of_snake - 1)
# 修改得分
pygame.display.update() for event in pygame.event.get():
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_q:
game_over = True
game_close = False
if event.key == pygame.K_p:
gameLoop() # 获取事件
for event in pygame.event.get():
if event.type == pygame.QUIT:
game_over = True
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_LEFT or event.key == pygame.K_a:
x1_change = -snake_block
y1_change = 0
elif event.key == pygame.K_RIGHT or event.key == pygame.K_d:
x1_change = snake_block
y1_change = 0
elif event.key == pygame.K_UP or event.key == pygame.K_w:
y1_change = -snake_block
x1_change = 0
elif event.key == pygame.K_DOWN or event.key == pygame.K_s:
y1_change = snake_block
x1_change = 0 if x1 >= dis_width or x1 < 0 or y1 >= dis_height or y1 < 0:
# 判断为游戏失败
game_close = True x1 += x1_change
y1 += y1_change dis.fill(blue)
pygame.draw.rect(dis, green, [foodx, foody, snake_block, snake_block])
snake_Head = []
snake_Head.append(x1)
snake_Head.append(y1)
snake_List.append(snake_Head)
# 贪吃蛇的移动
if len(snake_List) > Length_of_snake:
del snake_List[0] # 判断是否咬到自身
for x in snake_List[:-1]:
if x == snake_Head:
game_close = True # 画蛇出来
our_snake(snake_block, snake_List)
Your_score(Length_of_snake - 1) # 更新画面
pygame.display.update() # 判断蛇是否迟到食物
if foodx - 10 <= x1 <= foodx + 10 and foody - 10 <= y1 <= foody + 10:
foodx = round(random.randrange(snake_block, dis_width - snake_block, snake_block))
foody = round(random.randrange(snake_block, dis_height - snake_block, snake_block))
# print(foodx, foody)
Length_of_snake += 1 # 调节帧率来控制蛇的速度
clock.tick(snake_speed) pygame.quit()
quit() gameLoop()

代码讲解

使用的pygamerandom两个库。Pygame是一种流行的Python游戏开发库,它提供了许多功能,使开发人员可以轻松创建2D游戏。它具有良好的跨平台支持,可以在多个操作系统上运行。random库是用来获取随机数的,后面使用其获取食物的随机坐标。

import pygame
import random

初始化一些后期用到的参数,以及初始化pygame,创建clock时钟对象。

# 初始化pygame
pygame.init()
# 以RGB的形式定义颜色
white = (255, 255, 255)
yellow = (255, 255, 102)
black = (0, 0, 0)
red = (213, 50, 80)
green = (0, 255, 0)
blue = (50, 153, 213) # 设置窗体大小
dis_width = 800
dis_height = 600
dis = pygame.display.set_mode((dis_width, dis_height))
# 设置窗体的名称
pygame.display.set_caption('贪吃蛇 KevenDuan1.0') # 创建clock对象
clock = pygame.time.Clock() # 蛇的宽度
snake_block = 15
# 蛇的速度
snake_speed = 15 # 从系统库里获取字体
font_style = pygame.font.SysFont("bahnschrift", 25)
score_font = pygame.font.SysFont("comicsansms", 35)

定义得分函数,实际上就是在画面中添加字体。

def Your_score(score):
value = score_font.render("Score: " + str(score), True, yellow)
# 在主surface里添加字体surface
dis.blit(value, [0, 0])

定义蛇体函数,实际上就是遍历蛇身队列,依次在画面中画出矩形。

def our_snake(snake_block, snake_list):
for x in snake_list:
pygame.draw.rect(dis, black, [x[0], x[1], snake_block, snake_block])

定义文本函数,实际上就是在画面中显示文字。

def message(msg, color):
mesg = font_style.render(msg, True, color)
dis.blit(mesg, [dis_width // 2 - 180, dis_height // 2])

两个循环结构一个判断游戏是否结束一个判断游戏是否需要退出。

    while not game_over:
while game_close == True:
dis.fill(blue)
message("Game over, press p again or q quit!", red)
Your_score(Length_of_snake - 1)
# 修改得分
pygame.display.update() for event in pygame.event.get():
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_q:
game_over = True
game_close = False
if event.key == pygame.K_p:
gameLoop()

获取键盘事件改变蛇头方向

# 获取事件
for event in pygame.event.get():
if event.type == pygame.QUIT:
game_over = True
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_LEFT or event.key == pygame.K_a:
x1_change = -snake_block
y1_change = 0
elif event.key == pygame.K_RIGHT or event.key == pygame.K_d:
x1_change = snake_block
y1_change = 0
elif event.key == pygame.K_UP or event.key == pygame.K_w:
y1_change = -snake_block
x1_change = 0
elif event.key == pygame.K_DOWN or event.key == pygame.K_s:
y1_change = snake_block
x1_change = 0

蛇的移动是通过在队头添加一个新的坐标,然后删掉队尾元素实现的。队列的长度始终为蛇长变量的值。

		x1 += x1_change
y1 += y1_change dis.fill(blue)
pygame.draw.rect(dis, green, [foodx, foody, snake_block, snake_block])
snake_Head = []
snake_Head.append(x1)
snake_Head.append(y1)
snake_List.append(snake_Head)
# 贪吃蛇的移动
if len(snake_List) > Length_of_snake:
del snake_List[0] # 判断是否咬到自身
for x in snake_List[:-1]:
if x == snake_Head:
game_close = True # 画蛇出来
our_snake(snake_block, snake_List)
Your_score(Length_of_snake - 1) # 更新画面
pygame.display.update()

判断蛇头是否吃到食物,设置一定的范围,防止出现永远吃不到的bug

# 判断蛇是否吃到食物
if foodx - 10 <= x1 <= foodx + 10 and foody - 10 <= y1 <= foody + 10:
foodx = round(random.randrange(snake_block, dis_width - snake_block, snake_block))
foody = round(random.randrange(snake_block, dis_height - snake_block, snake_block))
# print(foodx, foody)
Length_of_snake += 1

增加菜单功能

通过查阅资料,发现pygame_menu提供了比较完善的游戏菜单功能。我通过pip下载后,查阅了官方样例代码,这个菜单确实是比较精美的,正是我所需要给游戏添加的东西。

import pygame
import pygame_menu pygame.init() surface = pygame.display.set_mode((600, 400)) def set_difficulty(value, difficulty):
# Do the job here !
print(1)
pass def start_the_game():
# Do the job here !
pass menu = pygame_menu.Menu('Welcome', 600, 400,
theme=pygame_menu.themes.THEME_BLUE) menu.add.text_input('Name :', default='kevenduan')
menu.add.selector('Difficulty :', [('Hard', 1), ('Easy', 2)], onchange=set_difficulty)
menu.add.button('Play', start_the_game)
menu.add.button('Quit', pygame_menu.events.EXIT)
menu.mainloop(surface)

最终版本



新功能

最终版本添加了游戏界面菜单,有查询得分记录的功能,以及可以选择两个难度。

查询分数通过列表记录后,通过排序来显示得分前5的分数。

难度的控制是由蛇的速度来控制,由于咬到自身就游戏失败有点坑,在这里删除了。

完整代码

import pygame
import random
import pygame_menu score = [0, 0, 0, 0, 0]
pygame.init()
# 以RGB的形式定义颜色
white = (255, 255, 255)
yellow = (255, 255, 102)
black = (0, 0, 0)
red = (213, 50, 80)
green = (0, 255, 0)
blue = (50, 153, 213) # 设置窗体大小
dis_width = 800
dis_height = 600
dis = pygame.display.set_mode((dis_width, dis_height))
pygame.display.set_caption('Snakes KevenDuan 1.0') # 创建clock对象
clock = pygame.time.Clock() # 蛇的宽度
snake_block = 15
# 蛇的速度
snake_speed = 15 # 从系统库里获取字体
font_style = pygame.font.SysFont("bahnschrift", 25)
score_font = pygame.font.SysFont("comicsansms", 35) def Your_score(score):
value = score_font.render("Score: " + str(score), True, yellow)
# 在主surface里添加字体surface
dis.blit(value, [0, 0]) def our_snake(snake_block, snake_list):
for x in snake_list:
pygame.draw.rect(dis, black, [x[0], x[1], snake_block, snake_block]) def message(msg, color):
mesg = font_style.render(msg, True, color)
dis.blit(mesg, [dis_width // 2 - 150, dis_height // 2]) def gameLoop():
game_over = False
game_close = False x1 = dis_width / 2
y1 = dis_height / 2 x1_change = 0
y1_change = 0 # 存放蛇的身体
snake_List = []
# 蛇的长度
Length_of_snake = 1 # 食物坐标随机生成
foodx = round(random.randrange(snake_block, dis_width - snake_block, snake_block))
foody = round(random.randrange(snake_block, dis_height - snake_block, snake_block))
print(foodx, foody) while not game_over:
while game_close == True:
dis.fill(blue)
score.append(Length_of_snake - 1)
message(f"Game Over! your score is {Length_of_snake - 1}", red)
Your_score(Length_of_snake - 1)
# 修改得分
pygame.display.update()
pygame.time.wait(2000)
# 显示出菜单
menu.mainloop(dis) # 获取事件
for event in pygame.event.get():
if event.type == pygame.QUIT:
game_over = True
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_LEFT or event.key == pygame.K_a:
x1_change = -snake_block
y1_change = 0
elif event.key == pygame.K_RIGHT or event.key == pygame.K_d:
x1_change = snake_block
y1_change = 0
elif event.key == pygame.K_UP or event.key == pygame.K_w:
y1_change = -snake_block
x1_change = 0
elif event.key == pygame.K_DOWN or event.key == pygame.K_s:
y1_change = snake_block
x1_change = 0 if x1 >= dis_width or x1 < 0 or y1 >= dis_height or y1 < 0:
# 判断为游戏失败
game_close = True x1 += x1_change
y1 += y1_change dis.fill(blue)
pygame.draw.rect(dis, green, [foodx, foody, snake_block, snake_block])
snake_Head = []
snake_Head.append(x1)
snake_Head.append(y1)
snake_List.append(snake_Head)
# 贪吃蛇的移动
if len(snake_List) > Length_of_snake:
del snake_List[0] # 判断是否咬到自身
# for x in snake_List[:-1]:
# if x == snake_Head:
# game_close = True # 画蛇出来
our_snake(snake_block, snake_List)
Your_score(Length_of_snake - 1) # 更新画面
pygame.display.update() # 判断蛇是否吃到食物
if foodx - 10 <= x1 <= foodx + 10 and foody - 10 <= y1 <= foody + 10:
foodx = round(random.randrange(snake_block, dis_width - snake_block, snake_block))
foody = round(random.randrange(snake_block, dis_height - snake_block, snake_block))
# print(foodx, foody)
Length_of_snake += 1 # 调节帧率来控制蛇的速度
clock.tick(snake_speed) pygame.quit()
quit() def set_difficulty(value, difficulty):
# Do the job here !
global snake_speed
if snake_speed == 15: snake_speed = 30
else: snake_speed = 15 def start_the_game():
# Do the job here !
gameLoop() def rank():
# flag = True
while True:
dis.fill(blue)
score.sort()
print(score)
mesg1 = font_style.render(f"Your highest score is {score[-1]}", True, yellow)
dis.blit(mesg1, [dis_width // 2 - 150, dis_height // 2 - 30])
mesg2 = font_style.render(f"Ranking of scores:{score[-1:-6:-1]}", True, yellow)
dis.blit(mesg2, [dis_width // 2 - 190, dis_height // 2 + 10]) pygame.display.update()
pygame.time.wait(3000)
menu.mainloop(dis)
# menu.add.button('Quit', pygame_menu.events.EXIT)
# for event in pygame.event.get():
# if event.type == pygame.QUIT:
# flag = True menu = pygame_menu.Menu('Welcome to my game', 800, 600,
theme=pygame_menu.themes.THEME_BLUE) menu.add.text_input('Author :', default='KevenDuan')
menu.add.selector('Difficulty :', [('Easy', 1), ('Hard', 2)], onchange=set_difficulty)
menu.add.button('Yours socres', rank)
menu.add.button('Play', start_the_game)
menu.add.button('Quit', pygame_menu.events.EXIT)
menu.mainloop(dis)

Python实现贪吃蛇大作战的更多相关文章

  1. Unity3D游戏贪吃蛇大作战源码休闲益智手机小游戏完整项目

    <贪吃蛇大作战>一款休闲竞技游戏,不仅比拼手速,更考验玩家的策略. 视频演示: http://player.youku.com/player.php/sid/XMzc5ODA2Njg1Ng ...

  2. 贪吃蛇大作战canvas实现(手机触屏操作)--地图逻辑

    //html部分 <!DOCTYPE html><html><head lang="en"> <meta charset="UT ...

  3. [Untiy]贪吃蛇大作战(五)——游戏主界面

    接着上一节: 4.AI蛇的设计 这里AI蛇大部分代码都可以参照主角的代码,我这里的实现其实还可以进行改进.基本原理就是蛇创建之后给蛇一个随机方向的单位向量,AI蛇的蛇头添加一个比蛇头大两三倍大小的碰撞 ...

  4. [Untiy]贪吃蛇大作战(四)——游戏主界面

    游戏主界面: 由于这个场景比较复杂,需要分几个部分实现: 1.游戏背景 首先我们的游戏场景上包括了一个大的背景图片,之外再包围一个红色的区域.中间的区域才是可活动的区域,周围通过碰撞检测盒来检测是否有 ...

  5. [Untiy]贪吃蛇大作战(三)——商店界面

    游戏商店界面: 实际的效果图如下: 要实现这个滑动,首先我们需要,一个内容显示区域,一个内容滚动区域,如下图: 其中ItemContent挂载的组件如下: 红框标注的地方是右方的滑动块. 然后Item ...

  6. [Untiy]贪吃蛇大作战(二)——规则界面

    游戏规则界面: 从界面上可以看出,一共有三个按钮,两个切换按钮和一个退出按钮. 一共三张规则图片Sprite,我们通过设置其是否为Active来控制显示,其控制脚本代码如下: using System ...

  7. [Untiy]贪吃蛇大作战(一)——开始界面

    前言: 刚学unity没多久吧(大概1个月多点),这是我自己做的除官网之外的第一个游戏demo,中间存在很多不足的地方,但是还是希望可以给需要的人提供一些思路和帮助,有问题的小伙伴可以找我一起探讨一起 ...

  8. 【python】10分钟教你用python打造贪吃蛇超详细教程

    10分钟教你用python打造贪吃蛇超详细教程 在家闲着没妹子约, 刚好最近又学了一下python,听说pygame挺好玩的.今天就在家研究一下, 弄了个贪吃蛇出来.希望大家喜欢. 先看程序效果: 0 ...

  9. 多线程的Python 教程--“贪吃蛇”

    本指南的里代码可以在这里下载:  threadworms.py ,或者从  GitHub.代码需要  Python 3 或 Python 2 ,同时也需要安装  Pygame . 点击查看大版本图片 ...

  10. 如何用python制作贪吃蛇以及AI版贪吃蛇

    用python制作普通贪吃蛇 哈喽,大家不知道是上午好还是中午好还是下午好还是晚上好! 很多人学习python,不知道从何学起.很多人学习python,掌握了基本语法过后,不知道在哪里寻找案例上手.很 ...

随机推荐

  1. ptp 时钟同步

    转载请注明出处: PTP(Precision Time Protocol)的功能可以帮助实现网络中各个节点的时钟同步,以提供更精确的时间参考. 作用: 时钟同步:通过PTP协议,在网络中不同节点之间实 ...

  2. 破局DevOps|8大北极星指标指引研发效能方向

    放弃那些动辄就上百个的研发度量指标吧,8大北极星指标指引你的研发效能方向,1个北极星指标公式让你清晰了解​公司研发效能现状. 每当研发效能/DevOps业务做规划的时候,有的人就会毫无头绪,不知道如何 ...

  3. Unity 性能优化之Shader分析处理函数ShaderUtil.HasProceduralInstancing: 深入解析与实用案例

    Unity 性能优化之Shader分析处理函数ShaderUtil.HasProceduralInstancing: 深入解析与实用案例 点击封面跳转到Unity国际版下载页面 简介 在Unity中, ...

  4. 万字+20张图剖析Spring启动时核心的12个步骤

    大家好,我是三友~~ 今天来扒一扒Spring在启动过程中核心的12个步骤 之所以来写这篇文章,主要是来填坑的 之前在三万字盘点Spring 9大核心基础功能这篇文章的末尾中给自己挖了一个坑,提了一嘴 ...

  5. 局域网内文件分享的简单方式:python - http.server

    在局域网条件下,利用Python自带的HTTP服务功能提供文件共享服务是相对比较简单便捷的方式之一. 一.现实需求及前提条件 1. 文件的服务端(文件分享者)与接收端(文件接收者)在一个局域网,接收端 ...

  6. Springboot中使用线程池的三种方式

    前言 多线程是每个程序员的噩梦,用得好可以提升效率很爽,用得不好就是埋汰的火葬场. 这里不深入介绍,主要是讲解一些标准用法,熟读唐诗三百首,不会作诗也会吟. 这里就介绍一下springboot中的多线 ...

  7. 【RocketMQ】消息的消费总结

    消费者从Broker拉取到消息之后,会将消息提交到线程池中进行消费,RocketMQ消息消费是批量进行的,如果一批消息的个数小于预先设置的批量消费大小,直接构建消费请求ConsumeRequest将消 ...

  8. Solution -「CSP-S 2020」函数调用

    Description 大家应该都读过题. Solution 赛后变摩托. 我们对每一个操作 \(3\) 连边建图,然后可以知道只是一个 \(\texttt{DAG}\). 考虑操作 \(2\),我们 ...

  9. Solution Set -「CF 1490」

    「CF 1490A」Dense Array Link. 显然不满足的 adjacent elements 之间一直加 \(\min\times2,\min\times4,\cdots,\min\tim ...

  10. Navicat连接MySQL,出现2059 - authentication plugin 'caching_sha2_password'的解决方法

    cmd(管理员)登录mysql执行如下 ALTER USER 'YOURUSERNAME'@'localhost' IDENTIFIED WITH mysql_native_password BY ' ...