200行Python代码实现贪吃蛇

话不多说,最后会给出全部的代码,也可以从这里Fork,正文开始;

目前实现的功能列表:

  1. 贪吃蛇的控制,通过上下左右方向键;
  2. 触碰到边缘、墙壁、自身则游戏结束;
  3. 接触到食物则食物消失,同时根据食物类型身体会变长;
  4. 目前长度显示;
  5. 暂停、死亡界面;

运行动图

代码片段分析

各个部分绘制的代码

# 游戏背景以及最下方用于显示文字的背景
def draw_background():
# white background
screen.fill(COLORS['white'])
pygame.draw.rect(screen,COLORS['black'],(-100,GAME_SIZE[1],3000,200),0) # 绘制墙壁
def draw_wall():
for xy in wall_list:
pygame.draw.rect(screen,COLORS['darkgray'],(xy[0]-WALL_WIDTH/2,xy[1]-WALL_WIDTH/2,WALL_WIDTH,WALL_HEIGHT),0) # 绘制蛇,包括头和身体
def draw_snake():
head = snake_list[0]
pygame.draw.circle(screen,COLORS['darkred'],(head[0],head[1]),int(SNAKE_WIDTH/2),0)
for xy in snake_list[1:]:
pygame.draw.rect(screen,COLORS['darkred'],(xy[0]-SNAKE_WIDTH/2,xy[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT),2) # 绘制食物
def draw_food():
for xyz in food_list:
pygame.draw.rect(screen,FOOD_COLORS[xyz[2]-1],(xyz[0]-FOOD_WIDTH/2,xyz[1]-FOOD_WIDTH/2,FOOD_WIDTH,FOOD_HEIGHT),0) # 绘制下方的身体长度记录
def draw_context():
txt = FONT_M.render('Snake length: '+str(len(snake_list)-1),True,COLORS['lightblue'])
x,y = 10,GAME_SIZE[1]+(int((SIZE[1]-GAME_SIZE[1])/2))
y = int(y-FONT_M.size('Count')[1]/2)
screen.blit(txt,(x,y)) # 绘制暂停界面
def draw_pause():
s = pygame.Surface(SIZE, pygame.SRCALPHA)
s.fill((255,255,255,220))
screen.blit(s, (0,0))
txt = FONT_M.render('PAUSE',True,COLORS['darkgray'])
x,y = SIZE[0]/2,SIZE[1]/2
x,y = int(x-FONT_M.size('PAUSE')[0]/2),int(y-FONT_M.size('PAUSE')[1]/2)
screen.blit(txt,(x,y)) # 绘制死亡界面
def draw_dead():
s = pygame.Surface(SIZE, pygame.SRCALPHA)
s.fill((255,255,255,240))
screen.blit(s, (0,0))
txt = FONT_M.render('YOU DEAD',True,COLORS['black'])
x,y = SIZE[0]/2,SIZE[1]/2
x,y = int(x-FONT_M.size('YOU DEAD')[0]/2),int(y-FONT_M.size('YOU DEAD')[1]/2)
screen.blit(txt,(x,y))

死亡与食物的碰撞检查

# 矩形覆盖检查作为碰撞检测,思路是取反,即取所有不会覆盖的情况的反即可
def rect_cover(rect1,rect2):
left1 = int(rect1[0])
right1 = int(rect1[0]+rect1[2])
up1 = int(rect1[1])
down1 = int(rect1[1]+rect1[3])
left2 = int(rect2[0])
right2 = int(rect2[0]+rect2[2])
up2 = int(rect2[1])
down2 = int(rect2[1]+rect2[3]) if not (right2<=left1 or left2>=right1 or down2<=up1 or up2>=down1):
return True
return False # 检查是否碰到食物
def check_food():
# 头与食物
first = snake_list[0]
snake_head_rect = (first[0]-SNAKE_WIDTH/2,first[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT)
for i in range(len(food_list)):
xyz = food_list[i]
food_rect = (xyz[0]-FOOD_WIDTH/2,xyz[1]-FOOD_WIDTH/2,FOOD_WIDTH,FOOD_HEIGHT)
if rect_cover(snake_head_rect,food_rect):
add_body(xyz[2])
del food_list[i]
return True
return False # 检查是否碰到边缘、墙壁或者自己的身体
def check_dead():
first = snake_list[0]
snake_head_rect = (first[0]-SNAKE_WIDTH/2,first[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT)
# 头与边缘
if first[0] < 0 or first[0] > GAME_SIZE[0] or first[1] < 0 or first[1] > GAME_SIZE[1]:
return True
# 头与墙壁
for xy in wall_list:
wall_rect = (xy[0]-WALL_WIDTH/2,xy[1]-WALL_WIDTH/2,WALL_WIDTH,WALL_HEIGHT)
if rect_cover(snake_head_rect,wall_rect):
return True
# 头与自身
for xy in snake_list[1:]:
body_rect = (xy[0]-SNAKE_WIDTH/2,xy[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT)
if rect_cover(snake_head_rect,body_rect):
return True
return False

更新食物以及增加蛇的身体长度

def add_food():
while(True):
xyz = [random.choice(X_LIST),random.choice(Y_LIST),random.choice([1,2,3,4])]
if xyz not in wall_list:
food_list.append(xyz)
break
def add_body(length=1):
for c in range(length):
# 尾巴加一节
last2,last1 = snake_list[-2],snake_list[-1]
if last2[0]==last1[0]: # 竖着的两段
if last2[1]>last1[1]: # 朝下
snake_list.append([last1[0],last1[1]-SNAKE_WIDTH])
else:
snake_list.append([last1[0],last1[1]+SNAKE_WIDTH])
else: # 横着的两段
if last2[0]>last1[0]: # 朝右
snake_list.append([last1[0]-SNAKE_WIDTH,last1[1]])
else:
snake_list.append([last1[0]+SNAKE_WIDTH,last1[1]])

蛇的自动移动

# 通过按键判断当前蛇的朝向
if event.key == K_LEFT:
if head in ['up','down']:
head = 'left'
elif event.key == K_RIGHT:
if head in ['up','down']:
head = 'right'
elif event.key == K_UP:
if head in ['left','right']:
head = 'up'
elif event.key == K_DOWN:
if head in ['left','right']:
head = 'down' # 通过朝向判断蛇的下一个位置
first = snake_list[0]
snake_list[1:] = snake_list[:-1]
if head == 'up':
snake_list[0] = [first[0],first[1]-SNAKE_WIDTH]
elif head == 'down':
snake_list[0] = [first[0],first[1]+SNAKE_WIDTH]
elif head == 'left':
snake_list[0] = [first[0]-SNAKE_WIDTH,first[1]]
elif head == 'right':
snake_list[0] = [first[0]+SNAKE_WIDTH,first[1]]

全部代码

import sys,random

import pygame
from pygame.color import THECOLORS as COLORS
from pygame.locals import * def draw_background():
# white background
screen.fill(COLORS['white'])
pygame.draw.rect(screen,COLORS['black'],(-100,GAME_SIZE[1],3000,200),0) def draw_wall():
for xy in wall_list:
pygame.draw.rect(screen,COLORS['darkgray'],(xy[0]-WALL_WIDTH/2,xy[1]-WALL_WIDTH/2,WALL_WIDTH,WALL_HEIGHT),0) def draw_snake():
head = snake_list[0]
pygame.draw.circle(screen,COLORS['darkred'],(head[0],head[1]),int(SNAKE_WIDTH/2),0)
for xy in snake_list[1:]:
pygame.draw.rect(screen,COLORS['darkred'],(xy[0]-SNAKE_WIDTH/2,xy[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT),2) def draw_food():
for xyz in food_list:
pygame.draw.rect(screen,FOOD_COLORS[xyz[2]-1],(xyz[0]-FOOD_WIDTH/2,xyz[1]-FOOD_WIDTH/2,FOOD_WIDTH,FOOD_HEIGHT),0) def draw_context():
txt = FONT_M.render('Snake length: '+str(len(snake_list)-1),True,COLORS['lightblue'])
x,y = 10,GAME_SIZE[1]+(int((SIZE[1]-GAME_SIZE[1])/2))
y = int(y-FONT_M.size('Count')[1]/2)
screen.blit(txt,(x,y)) def draw_pause():
s = pygame.Surface(SIZE, pygame.SRCALPHA)
s.fill((255,255,255,220))
screen.blit(s, (0,0))
txt = FONT_M.render('PAUSE',True,COLORS['darkgray'])
x,y = SIZE[0]/2,SIZE[1]/2
x,y = int(x-FONT_M.size('PAUSE')[0]/2),int(y-FONT_M.size('PAUSE')[1]/2)
screen.blit(txt,(x,y)) def draw_dead():
s = pygame.Surface(SIZE, pygame.SRCALPHA)
s.fill((255,255,255,240))
screen.blit(s, (0,0))
txt = FONT_M.render('YOU DEAD',True,COLORS['black'])
x,y = SIZE[0]/2,SIZE[1]/2
x,y = int(x-FONT_M.size('YOU DEAD')[0]/2),int(y-FONT_M.size('YOU DEAD')[1]/2)
screen.blit(txt,(x,y)) def rect_cover(rect1,rect2):
left1 = int(rect1[0])
right1 = int(rect1[0]+rect1[2])
up1 = int(rect1[1])
down1 = int(rect1[1]+rect1[3])
left2 = int(rect2[0])
right2 = int(rect2[0]+rect2[2])
up2 = int(rect2[1])
down2 = int(rect2[1]+rect2[3]) if not (right2<=left1 or left2>=right1 or down2<=up1 or up2>=down1):
return True
return False def add_food():
while(True):
xyz = [random.choice(X_LIST),random.choice(Y_LIST),random.choice([1,2,3,4])]
if xyz not in wall_list:
food_list.append(xyz)
break
def add_body(length=1):
for c in range(length):
# 尾巴加一节
last2,last1 = snake_list[-2],snake_list[-1]
if last2[0]==last1[0]: # 竖着的两段
if last2[1]>last1[1]: # 朝下
snake_list.append([last1[0],last1[1]-SNAKE_WIDTH])
else:
snake_list.append([last1[0],last1[1]+SNAKE_WIDTH])
else: # 横着的两段
if last2[0]>last1[0]: # 朝右
snake_list.append([last1[0]-SNAKE_WIDTH,last1[1]])
else:
snake_list.append([last1[0]+SNAKE_WIDTH,last1[1]]) def check_food():
# 头与食物
first = snake_list[0]
snake_head_rect = (first[0]-SNAKE_WIDTH/2,first[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT)
for i in range(len(food_list)):
xyz = food_list[i]
food_rect = (xyz[0]-FOOD_WIDTH/2,xyz[1]-FOOD_WIDTH/2,FOOD_WIDTH,FOOD_HEIGHT)
if rect_cover(snake_head_rect,food_rect):
add_body(xyz[2])
del food_list[i]
return True
return False def check_dead():
first = snake_list[0]
snake_head_rect = (first[0]-SNAKE_WIDTH/2,first[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT)
# 头与边缘
if first[0] < 0 or first[0] > GAME_SIZE[0] or first[1] < 0 or first[1] > GAME_SIZE[1]:
return True
# 头与墙壁
for xy in wall_list:
wall_rect = (xy[0]-WALL_WIDTH/2,xy[1]-WALL_WIDTH/2,WALL_WIDTH,WALL_HEIGHT)
if rect_cover(snake_head_rect,wall_rect):
return True
# 头与自身
for xy in snake_list[1:]:
body_rect = (xy[0]-SNAKE_WIDTH/2,xy[1]-SNAKE_WIDTH/2,SNAKE_WIDTH,SNAKE_HEIGHT)
if rect_cover(snake_head_rect,body_rect):
return True
return False if __name__ == "__main__":
# init pygame
pygame.init() # contant
GAME_SIZE = [900,900]
SIZE = [GAME_SIZE[0],GAME_SIZE[1]+100]
FONT_S = pygame.font.SysFont('Times', 50)
FONT_M = pygame.font.SysFont('Times', 90)
DIRECTION = ['up','right','down','left']
X_LIST = [x for x in range(GAME_SIZE[0])]
Y_LIST = [y for y in range(GAME_SIZE[1])]
FOOD_COLORS = ((46,139,87),(199,21,133),(25,25,112),(255,215,0)) # wall
wall_list = [[100,200],[600,500],[350,200],[500,800]]
WALL_WIDTH,WALL_HEIGHT = 30,30 # food
food_list = [(150,200,1),(300,500,1),(740,542,1),(300,600,1),(700,600,1)]
FOOD_WIDTH,FOOD_HEIGHT = 14,14 # create screen 500*500
screen = pygame.display.set_mode(SIZE) # variable parameter
snake_list = [[100+12*4,100],[100+12*3,100],[100+12*2,100],[100+12*1,100],[100,100]]
SNAKE_WIDTH,SNAKE_HEIGHT = 12,12
snake_v = 0
count_time = 0 # level
frame = 0.05
level = 1 # main loop
running = True
pause = False
dead = False
head = 'right'
while running:
for event in pygame.event.get():
if event.type == pygame.QUIT:
running = False
break
elif event.type == pygame.MOUSEBUTTONDOWN:
pause = not pause
elif event.type == pygame.KEYUP:
if event.key == K_LEFT:
if head in ['up','down']:
head = 'left'
elif event.key == K_RIGHT:
if head in ['up','down']:
head = 'right'
elif event.key == K_UP:
if head in ['left','right']:
head = 'up'
elif event.key == K_DOWN:
if head in ['left','right']:
head = 'down' # update data
if not pause and not dead:
count_time += frame*level
first = snake_list[0]
snake_list[1:] = snake_list[:-1]
if head == 'up':
snake_list[0] = [first[0],first[1]-SNAKE_WIDTH]
elif head == 'down':
snake_list[0] = [first[0],first[1]+SNAKE_WIDTH]
elif head == 'left':
snake_list[0] = [first[0]-SNAKE_WIDTH,first[1]]
elif head == 'right':
snake_list[0] = [first[0]+SNAKE_WIDTH,first[1]] # background
draw_background()
# tunnel
draw_wall()
# choose item
draw_snake()
# food
draw_food()
# point
draw_context()
# pause
if not dead and pause:
draw_pause()
# dead
if dead:
draw_dead()
# flip
pygame.display.flip() # pause 20ms
pygame.time.delay(int(frame/level*1000)) # check win or not
dead = check_dead() if check_food():
add_food() pygame.quit()

再贴一下这个仓库的链接

小游戏集合仓库

最后

大家可以到我的Github上看看有没有其他需要的东西,目前主要是自己做的机器学习项目、Python各种脚本工具、有意思的小项目以及Follow的大佬、Fork的项目等:

https://github.com/NemoHoHaloAi

200行PYTHON代码实现贪吃蛇的更多相关文章

  1. 20行JS代码实现贪吃蛇

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  2. 200行Python代码实现2048

    200行Python代码实现2048 一.实验说明 1. 环境登录 无需密码自动登录,系统用户名shiyanlou 2. 环境介绍 本实验环境采用带桌面的Ubuntu Linux环境,实验中会用到桌面 ...

  3. 用200行Python代码“换脸”

    介绍 本文将介绍如何编写一个只有200行的Python脚本,为两张肖像照上人物的“换脸”. 这个过程可分为四步: 检测面部标记. 旋转.缩放和转换第二张图像,使之与第一张图像相适应. 调整第二张图像的 ...

  4. python 之路,200行Python代码写了个打飞机游戏!

    早就知道pygame模块,就是没怎么深入研究过,恰逢这周未没约到妹子,只能自己在家玩自己啦,一时兴起,花了几个小时写了个打飞机程序. 很有意思,跟大家分享下. 先看一下项目结构 "" ...

  5. 小游戏:200行python代码手写2048

    #-*- coding: utf-8 -*- import curses from random import randrange, choice from collections import de ...

  6. (转)python 之路,200行Python代码写了个打飞机游戏!

    原文:https://www.cnblogs.com/alex3714/p/7966656.html

  7. Python制作AI贪吃蛇,很多很多细节、思路都写下来了!

    前提:本文实现AI贪吃蛇自行对战,加上人机对战,读者可再次基础上自行添加电脑VS电脑和玩家VS玩家(其实把人机对战写完,这2个都没什么了,思路都一样) 实现效果: 很多人学习python,不知道从何学 ...

  8. 20行Python代码爬取王者荣耀全英雄皮肤

    引言王者荣耀大家都玩过吧,没玩过的也应该听说过,作为时下最火的手机MOBA游戏,咳咳,好像跑题了.我们今天的重点是爬取王者荣耀所有英雄的所有皮肤,而且仅仅使用20行Python代码即可完成. 准备工作 ...

  9. 一个 11 行 Python 代码实现的神经网络

    一个 11 行 Python 代码实现的神经网络 2015/12/02 · 实践项目 · 15 评论· 神经网络 分享到:18 本文由 伯乐在线 - 耶鲁怕冷 翻译,Namco 校稿.未经许可,禁止转 ...

随机推荐

  1. 用 git 钩子,检测代码规范性(eslint、standard)

    最终实现效果说明:用 git commit 提交代码之前,利用 pre-commit git 钩子,实现代码规范检测(eslint.standard 规范),符合规范之后才可以提交到 git 仓库.这 ...

  2. vue+element tree(树形控件)组件(1)

    最近做了第一个组内可以使用的组件,虽然是最简版,也废了不少力.各位前辈帮我解决问题,才勉强搞定.让我来记录这个树形组件的编写过程和期间用到的知识点. 首先说说需求,就是点击出现弹窗+蒙板,弹窗内容是一 ...

  3. JDBC概述及编程步骤详解

    目录 一.JDBC概述 二.JDBC编程步骤(以MySQL为例) 1.导入jar包 2.加载数据库驱动 3.通过DriverManager获得Connection对象 3.定义SQL语句 4.利用Co ...

  4. Vue请求第三方接口跨域最终解决办法!2020最终版!

    废话少说,再百度的近三个小时尝试了近10种方法无解后,终于皇天不负有心人! 这个vue axios 跨域问题被我解决了! 需求:请求客户端ip地址获取客户ip,再根据ip获取用户位置 工具:Vue,a ...

  5. 微服务架构-Gradle下载安装配置教程

    一.开发条件 JDK8下载地址:https://www.oracle.com/java/technologies/javase-jdk8-downloads.html Eclipse下载地址:http ...

  6. angular 动态绑定class 写法

    [ngClass]="{'aaa':true,'bbb':!true}" 这个是正确的   [ngClass]="{ true ? 'aaa':'bbb'}" ...

  7. DEX文件解析---2、Dex文件checksum(校验和)解析

    一.checksum介绍     checksum(校验和)是DEX位于文件头部的一个信息,用来判断DEX文件是否损坏或者被篡改,它位于头部的0x08偏移地址处,占用4个字节,采用小端序存储.     ...

  8. 前端Tips#6 - 在 async iterator 上使用 for-await-of 语法糖

    视频讲解 前往原文 前端Tips 专栏#6,点击观看 文字讲解 本期主要是讲解如何使用 for-await-of 语法糖进行异步操作迭代,让组织异步操作的代码更加简洁易读. 1.场景简述 以下代码中的 ...

  9. 外部配置属性值是如何被绑定到XxxProperties类属性上的?--SpringBoot源码(五)

    注:该源码分析对应SpringBoot版本为2.1.0.RELEASE 1 前言 本篇接 SpringBoot是如何实现自动配置的?--SpringBoot源码(四) 温故而知新,我们来简单回顾一下上 ...

  10. django验证码框架captcha

    1.安装 2.在settings.py 安装app中添加 3.添加url 4.运行makemigrations和migrate 5.运用 在form表单中定义 view中返回form表单 在前端htm ...