贪吃蛇python小白_面向 python 小白的贪吃蛇游戏

代码和教程详见微信公众号:Python高效编程

效果图

图片

代码和教程详见微信公众号:Python高效编程

文字部分:

引言

作为python 小白,总是觉得自己要做好百分之二百的准备,才能开始写程序。以至于常常整天在那看各种语法教程,学了几个月还是只会print('hello world')。

这样做效率太低,正确的做法,是到身边找问题,然后编程实现。比如说,我学了高等数学,我是不是应该考虑下如何去用编程实现求导或者积分操作,如果想不出怎么办,是不是应该 baidu 一下,别人是如何实现数值积分或是符号积分的。我们每天买东西都要用到加减甚至乘除,那么我是否能编写个简单的计算器,如果命令行太丑的话,我是否能够快速地学一学 pyqt5或是其他 gui来实现精致些的应用程序。凡事用编程思维考虑一下,对于从编程小白进阶为编程入门是大有裨益的。

小时候,我们或多或少会沉迷于一款经久不衰的游戏------贪吃蛇。或许我们玩过各式各样的贪吃蛇游戏,却没有自己动手编写属于自己的贪吃蛇游戏。今天就让我们走进贪吃蛇的世界,用 python 实现简易版的贪吃蛇游戏。

游戏简介

首先是游戏效果图:效果图

用户通过操控贪吃蛇,去吃到尽可能多的食物。其中贪吃蛇不能碰到墙壁,也不能咬到自身。

本教程借助 pygame实现游戏界面,所以下面稍稍介绍一下 pygame的安装,用法就在下面连同函数一起讲了:

安装:

1 pip install -U pygame

接下来让我们介绍下实现贪吃蛇的关键逻辑:

贪吃蛇的身体是由list构成的,list中每一个元组代表贪吃蛇在棋盘上的坐标,我们只需在这些位置画上图案,就能制作出一条圆滚滚的贪吃蛇来。但是如果想让贪吃蛇活蹦乱跳,我们就要写一个move函数。

那么贪吃蛇怎么移动呢?

如果贪吃蛇没吃到食物,那么我们就删除list中最后一个坐标,再在蛇头部分插入新的位置。如何确定新的位置呢,我们就要设定贪吃蛇移动的方向(x,y),将原蛇头位置的坐标在移动方向上进行加减操作。这样贪吃蛇就实现了向前移动的目标。如果贪吃蛇恰好吃到了食物,唯一的不同就是不需要删除贪吃蛇尾部的元素。其中需要注意的是,贪吃蛇不能朝着当前移动方向的反方向移动。体现在代码中,就是当前方向与改变方向的乘积不能为负值。

那么如何知道贪吃蛇吃到了食物呢?

如果贪吃蛇蛇头的坐标与食物的坐标重合的话,贪吃蛇就吃到了食物。如果贪吃蛇吃到了食物,就在棋盘上随机更新食物。如果随机生成的食物的坐标,恰好与贪吃蛇的位置重合的话,就继续随机产生坐标,直到确保与贪吃蛇的坐标不同的时候。

那么如何知道游戏失败了呢?

如果贪吃蛇蛇头的坐标与边框的坐标重合的话,蛇卒。如果贪吃蛇各个部分的坐标有重合的话,就说明贪吃蛇咬到了自己,游戏结束。

接下来是各个部分的具体代码实现:

下图为主要需要的几个函数:代码结构

首先__init__初始化贪吃蛇的位置,初始方向竖直向上。toward函数用于改变贪吃蛇的方向,(x,y)分别表示蛇头在水平和竖直方向的朝向。朝左x=-1,朝右x=1,朝上y=-1,朝下y=1。move函数,使用标志enlarge来判断蛇是否吃到了食物,并进行相应的操作。eat_food函数判断蛇是否吃到食物,吃到的话,分数加 100,并返回True。toward函数,用于改变蛇头的方向,但如果改变方向与当前方向相反,就什么操作都不执行。draw函数用于画出贪吃蛇的模样,蛇头是略大一点的红心⚪,蛇身是小一点的黄心⚪。

我们怎么画出这条蛇呢?这就要借助函数pygame.draw.circle,这个函数的主要参数有screen:就是你要在其中画出贪吃蛇的游戏界面,color:图案的颜色(RGB), position:图案在屏幕上的位置, radius:⚪的半径,width:内部填色的大小,如果为零,图案就是空心圆;如果与半径大小相同,图案就是实心圆。

下面是贪吃蛇的代码部分,大家可以结合注释阅读:

1# 贪吃蛇

2class Snack(object):

3    def __init__(self):

4        # self.item = [(3, 25), (2, 25), (1, 25), (1,24), (1,23),

5        # (1,22), (1,21), (1,20), (1,19), (1,18), (1,17), (1,16)]

6        # x 水平方向 y 竖直方向

7        # 初始方向竖直向上

8        self.item = [(3, 25), (2, 25), (1, 25), (1, 24), ]

9        self.x = 0

10        self.y = -1

11

12    def move(self, enlarge):

13        # enlarge 标记贪吃蛇有没有吃到食物

14        if not enlarge:

15            # 吃到食物删除尾部元素

16            self.item.pop()

17        # 新蛇头的坐标为旧蛇头坐标加上移动方向的位移

18        head = (self.item[0][0] + self.x, self.item[0][1] + self.y)

19        # 将新的蛇头坐标插入在 list 最前面

20        self.item.insert(0, head)

21

22    def eat_food(self, food):

23        global score

24        # snack_x,snack_y 蛇头坐标

25        # food_x, food_y 食物坐标

26        snack_x, snack_y = self.item[0]

27        food_x, food_y = food.item

28        # 比较蛇头坐标与食物坐标

29        if (food_x == snack_x) and (food_y == snack_y):

30            score += 100

31            return 1

32        else:

33            return 0

34

35    def toward(self, x, y):

36        # 改变蛇头朝向

37        if self.x * x >= 0 and self.y * y >= 0:

38            self.x = x

39            self.y = y

40

41    def get_head(self):

42        # 获取蛇头坐标

43        return self.item[0]

44

45    def draw(self, screen):

46        # 画出贪吃蛇

47        # 蛇头为半径为 15 的红色实心圆

48        radius = 15

49        width = 15

50        # i:1---34   j:1---25

51        color = 255, 0, 0

52        # position 为图形的坐标

53        position = 10 + 20 * self.item[0][0], 10 + 20 * self.item[0][1]

54        pygame.draw.circle(screen, color, position, radius, width)

55        # 蛇身为半径为 10 的黄色实心圆

56        radius = 10

57        width = 10

58        color = 255, 255, 0

59        for i, j in self.item[1:]:

60            position = 10 + 20 * i, 10 + 20 * j

61            pygame.draw.circle(screen, color, position, radius, width)

其次是食物模块:

np.random.randint用于产生边界之内的坐标,如果与贪吃蛇的坐标重合,那么就继续生成新的随机坐标。

1# 食物

2class Food(object):

3    def __init__(self):

4        self.item = (4, 5)

5

6    # 画出食物

7    def _draw(self, screen, i, j):

8        color = 255, 0, 255

9        radius = 10

10        width = 10

11        # i:1---34   j:1---25

12        position = 10 + 20 * i, 10 + 20 * j

13        # 画出半径为 10 的粉色实心圆

14        pygame.draw.circle(screen, color, position, radius, width)

15

16    # 随机产生食物

17    def update(self, screen, enlarge, snack):

18        if enlarge:

19            self.item = np.random.randint(1, BOARDWIDTH - 2), np.random.randint(1, BOARDHEIGHT - 2)

20            while self.item in snack.item:

21                self.item = np.random.randint(1, BOARDWIDTH - 2), np.random.randint(1, BOARDHEIGHT - 2)

22        self._draw(screen, self.item[0], self.item[1])

然后是init_board函数:

board_width、board_height分别为游戏界面的宽度和高度,根据计算得出边框占据的位置,然后打印出正方形来。pygame.draw.rect和pygame.draw.circle用法类似,区别就是rect四个参数分别为screen:屏幕,color:颜色,pos:横坐标 x,纵坐标 y,矩形的长,矩形的宽。这里我设置矩形长宽都为 20 。width和circle中width用法相同,都是填充大小的意思。

1# 初始界面

2def init_board(screen):

3    board_width = BOARDWIDTH

4    board_height = BOARDHEIGHT

5    color = 10, 255, 255

6    width = 0

7    # width:x, height:y

8    # 左右边框占用了 X: 0 35*20

9    for i in range(board_width):

10        pos = i * 20, 0, 20, 20

11        pygame.draw.rect(screen, color, pos, width)

12        pos = i * 20, (board_height - 1) * 20, 20, 20

13        pygame.draw.rect(screen, color, pos, width)

14    # 上下边框占用了 Y: 0 26*20

15    for i in range(board_height - 1):

16        pos = 0, 20 + i * 20, 20, 20

17        pygame.draw.rect(screen, color, pos, width)

18        pos = (board_width - 1) * 20, 20 + i * 20, 20, 20

19        pygame.draw.rect(screen, color, pos, width)

接着是game_over模块:

如何判断谁咬到自身呢?可以利用python内置数据结构set:set这种数据结构中不能有重复元素。如果将list变成set之后,长度变短了,就说明list中有重复元素,即贪吃蛇咬到自己了。

1# 游戏失败

2def game_over(snack):

3    broad_x, broad_y = snack.get_head()

4    flag = 0

5    old = len(snack.item)

6    new = len(set(snack.item))

7    # 游戏失败的两种可能

8    # 咬到自身

9    if new < old:

10        flag = 1

11    # 撞到边框

12    if broad_x == 0 or broad_x == BOARDWIDTH - 1:

13        flag = 1

14    if broad_y == 0 or broad_y == BOARDHEIGHT - 1:

15        flag = 1

16

17    if flag:

18        return True

19    else:

20        return False

接下来是游戏初始化模块:

使用pygame模块需要使用pygame.init进行初始化。pygame.display.set_mode用来设置游戏界面的大小。pygame.display.set_caption用来显示游戏标题。

1# 游戏初始化

2def game_init():

3    # pygame 初始化

4    pygame.init()

5    # 设置游戏界面大小

6    screen = pygame.display.set_mode((BOARDWIDTH * 20, BOARDHEIGHT * 20))

7    # 设置游戏标题

8    pygame.display.set_caption('贪吃蛇游戏')

9    # sound = pygame.mixer.Sound(AUDIONAME)

10    # channel = pygame.mixer.find_channel(True)

11    # channel.play(sound)

12    return screen

最后是游戏主函数:

首先实例化贪吃蛇和食物。其次设置字体为SimHei,如果使用默认字体对中文的支持很不好。其次显示游戏界面,判断游戏是否失败。如果失败的话,就打印GAME OVER。否则就一直执行主函数。其中        pygame.event.get从队列中获取事件,也就是说必须先获取事件,才能得到用户的键盘输入和其他操作,screen.fill用于填充屏幕,pygame.key.get_pressed用于获取用户的键盘输入,pygame.display.update用来刷新到之前的图案,time.sleep用于控制刷新的频率。

1# 开始游戏

2def game(screen):

3    snack = Snack()

4    food = Food()

5    # 设置中文字体和大小

6    font = pygame.font.SysFont('SimHei', 20)

7    is_fail = 0

8    while True:

9        for event in pygame.event.get():

10            if event.type == QUIT:

11                exit()

12        # 填充屏幕

13        screen.fill((0, 0, 100))

14        init_board(screen=screen)

15        # 获得用户按键命令

16        keys = pygame.key.get_pressed()

17        press(keys, snack)

18        # 游戏失败打印提示

19        if is_fail:

20            font2 = pygame.font.Font(None, 40)

21            print_text(screen, font, 0, 0, text)

22            print_text(screen, font2, 400, 200, "GAME OVER")

23        # 游戏主进程

24        if not is_fail:

25            enlarge = snack.eat_food(food)

26            text = u"score: {}  更多精彩关注微信公众号:python高效编程".format(score)

27            print_text(screen, font, 0, 0, text)

28            food.update(screen, enlarge, snack)

29            snack.move(enlarge)

30            is_fail = game_over(snack=snack)

31            snack.draw(screen)

32        # 游戏刷新

33        pygame.display.update()

34        time.sleep(0.1)

好了,我们的贪吃蛇教程就这样结束了,其他零碎的知识点都在源码中。大家可以自己尝试编写自己的第一个贪吃蛇游戏了,还可以给自己的贪吃蛇扩展各种各样的功能。比如一边播放音乐,一边开始游戏,或者编写个更加美观的贪吃蛇界面。源码在微信公众号:Python高效编程。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/470250.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Redux学习(二)——封装connect函数

一、自定义connect函数 connect.js: import {PureComponent} from "react"; import store from "../store"; export function connect(mapStateToProps, mapDispatchToProps) {return function enhanceHOC(WrappedComponent) {return class extends PureC…

python ssh实现_SSH协议的Python实现paramiko

paramiko安装SSH是一个协议&#xff0c;paramiko 是一个Python 的库&#xff0c;该库支持sshv2协议&#xff0c;实现了对远程服务器执行操作安装命令&#xff1a;pip3 install paramikoSSHClient类与SFTPClient类paramiko包含两个核心组建&#xff0c;分别是SSHClient和SFTPCli…

C++学习10 static静态成员变量和静态成员函数

一般情况下&#xff0c;如果有N个同类的对象&#xff0c;那么每一个对象都分别有自己的成员变量&#xff0c;不同对象的成员变量各自有值&#xff0c;互不相干。但是有时我们希望有某一个或几个成员变量为所有对象共有&#xff0c;这样可以实现数据共享。 可以使用全局变量来达…

Redux学习(三)——redux-saga的使用、编写中间件函数、Reducer文件拆分

一、redux-devtools 我们之前讲过&#xff0c;redux可以方便的让我们对状态进行跟踪和调试&#xff0c;那么如何做到呢&#xff1f; redux官网为我们提供了redux-devtools的工具&#xff1b;利用这个工具&#xff0c;我们可以知道每次状态是如何被修改的&#xff0c;修改前后…

python 存储图片 alpha_保存时Matplotlib图形面颜色alpha(背景色、透明度)

前一个问题是使用savefig()保存屏幕上显示的相同的面颜色(背景色)&#xff0c;即&#xff1a;fig plt.figure()fig.patch.set_facecolor(blue)fig.savefig(foo.png, facecolorfig.get_facecolor())(使用savefig()需要重新指定背景色。)fig.patch.set_alpha(0.5)我找不到一种方…

【JavaScript代码实现四】获取和设置 cookie

1 // 创建cookie2 function setCookie(name, value, expires, path, domain, secure) {3 var cookieText encodeURIComponent(name) encodeURIComponent(value);4 if (expires instanceof Date) {5 cookieText ; expires expires; 6 }7 if…

react-router的使用(一)——URL的hash、HTML5的history、Router的基本使用

一、阶段一&#xff1a;后端路由阶段 早期的网站开发整个HTML页面是由服务器来渲染的. 服务器直接生产渲染好对应的HTML页面, 返回给客户端进行展示. 但是, 一个网站, 这么多页面服务器如何处理呢? 一个页面有自己对应的网址, 也就是URL.URL会发送到服务器, 服务器会通过正…

Linux中source是什么指令?

命令用法&#xff1a; source FileName作用&#xff1a;在当前bash环境下读取并执行FileName中的命令。 注&#xff1a;该命令通常用命令“.”来替代。 如&#xff1a;source /etc/profile 与 . /etc/profile 是等效的。 注意&#xff1a;source命令与shell scripts的区别是&a…

react-router的使用(二)——NavLink的使用、Switch的作用、Redirect

一、NavLink的使用 需求&#xff1a;路径选中时&#xff0c;对应的a元素变为红色 这个时候&#xff0c;我们要使用NavLink组件来替代Link组件&#xff1a; activeStyle&#xff1a;活跃时&#xff08;匹配时&#xff09;的样式&#xff1b;activeClassName&#xff1a;活跃时…

群晖ffmpeg_群晖Video station支持DTS和EAC3

群晖video station这个套件现在经过群晖的打磨&#xff0c;现在还是不错的&#xff0c;支持硬件解码和蓝光等多媒体播放&#xff0c;比起PLEX和EMBY动辄好几百的会员费&#xff0c;这个免费的用起来还真香&#xff0c;但是因为种种小问题需要解决了&#xff0c;才能好用&#x…

Redis 3.0.2集群搭建以及相关问题汇总

Redis3 正式支持了 cluster&#xff0c;是为了解决构建redis集群时的诸多不便&#xff08;1&#xff09;像操作单个redis一样操作key&#xff0c;不用操心key在哪个节点上&#xff08;2&#xff09;在线动态添加、删除redis节点&#xff0c;不用停止服务&#xff08;3&#xff…

react-router的使用(三)——路由的嵌套

一、路由的嵌套 在开发中&#xff0c;路由之间是存在嵌套关系的。 这里我们假设about页面中有三个页面内容&#xff1a; 企业历史、企业文化和联系我们&#xff1b;点击不同的链接可以跳转到不同的地方&#xff0c;显示不同的内容&#xff1b; 二、手动路由跳转 目前我们实现…

如何阅读一本书 pdf_《如何阅读一本书》:一本书,四个层次,看阅读小白如何逆袭?...

“读书不是为了雄辩和驳斥&#xff0c;也不是为了轻信和盲从&#xff0c;而是为了思考和权衡。”这是培根的一句名言&#xff0c;我们都曾经被这样的读书警句激励的斗志昂扬&#xff0c;于是立下目标一年或是一个月要读多少本书&#xff0c;结果发现一切是徒劳。你是否曾经斗志…

React Hooks的使用(一)——useState、useEffect解析

一、为什么需要Hook? Hook 是 React 16.8 的新增特性&#xff0c;它可以让我们在不编写class的情况下使用state以及其他的React特性&#xff08;比如生命周期&#xff09;。 我们先来思考一下class组件相对于函数式组件有什么优势&#xff1f;比较常见的是下面的优势&#xf…

python长沙_长沙python

本文转自量子位(ID:QbitAI) 边策 鱼羊 发自 凹非寺 量子位 报道 | 公众号 QbitAI 只用99行代码&#xff0c;你也可以像《冰雪奇缘》里的艾莎公主一样拥有冰雪魔法。虽然你不能在现实世界中肆意变出魔法&#xff0c;但却能在计算机的虚拟世界挥洒特效。或许你不知道&#xff0c;…

sed

sed -n 10p text.txt 输出第10行转载于:https://www.cnblogs.com/ssyuxue/p/5804857.html

React Hooks的使用(二)——useContext、useReducer、useCallback、useMemo解析

一、useContext的使用 在之前的开发中&#xff0c;我们要在组件中使用共享的Context有两种方式&#xff1a; 第一种方式&#xff1a;类组件可以通过 类名.contextType MyContext方式&#xff0c;在类中获取context&#xff1b; 第二种方式&#xff1a;多个Context或者在函数…

凸多边形面积_C++计算任意多边形的面积

任意多边形的面积计算_拾忆楓灵的博客-CSDN博客​blog.csdn.net计算任意多边形的面积 - tenos - 博客园​www.cnblogs.com完美解决计算3D空间任意多边形面积_Studiouss的博客-CSDN博客​blog.csdn.net求多边形面积公式&#xff08;已知顶点坐标&#xff09;_扬帆起航-CSDN博客​…

springContext

方法一 package com.hsh.utils; import org.springframework.beans.BeansException;import org.springframework.context.ApplicationContext;import org.springframework.context.ApplicationContextAware;import org.springframework.stereotype.Component;Componentpublic c…

React Hooks的使用(三)——useRef、useImperativeHandle、useLayoutEffect解析、自定义Hook

一、useRef useRef返回一个ref对象&#xff0c;返回的ref对象再组件的整个生命周期保持不变。 最常用的ref是两种用法&#xff1a; 用法一&#xff1a;引入DOM&#xff08;或者组件&#xff0c;但是需要是class组件&#xff09;元素&#xff1b; 案例一&#xff1a;引用DOM …