本文基于 Windows 环境开发,适合 Python 新手

下面我们就一起用 Python 实现一个简单有趣的命令行贪吃蛇小游戏,启动命令:
.私信小编01即可获取大量Python学习教程

本文包含设计和讲解,整体分为两个部分:第一部分是关于 Python 命令行图形化库curses接着是snake相关代码。
Python 已经内置了 curses 库,但是对于 Windows 操作系统我们需要安装一个补丁以进行适配。
Windows 下安装补全包:
pip install windows-curses
curses 是一个应用广泛的图形函数库,可以在终端内绘制简单的用户界面。
在这里我们只进行简单的介绍,只学习贪吃蛇需要的功能
如果您已经接触过 curses,请跳过此部分内容。
Python 内置了 curses 库,其使用方法非常简单,以下脚本可以显示出当前按键对应编号:
# 导入必须的库importcursesimporttime# 初始化命令行界面,返回的 stdscr 为窗口对象,表示命令行界面stdscr = curses.initscr()# 使用 noecho 方法关闭命令行回显curses.noecho()# 使用 nodelay 方法让 getch 为非阻塞等待stdscr.nodelay(True)whileTrue:# 清除 stdscr 窗口的内容(清除残留的符号)stdscr.erase()# 获取用户输入并放回对应按键的编号# 非阻塞等待模式下没有输入则返回 -1key = stdscr.getch()# 在 stdscr 的第一行第三列显示文字stdscr.addstr(1,3,"Hello GitHub.")# 在 stdscr 的第二行第三列显示文字stdscr.addstr(2,3,"Key: %d"% key)# 刷新窗口,让刚才的 addstr 生效stdscr.refresh()# 等待 0.1s 给用户足够反应时间查看文字time.sleep(0.1)

您也可以尝试把 nodelay 改为 nodelay 后再次运行,这时候程序会阻塞在 stdscr.getch 只有当您按下按键后才会继续执行。
您也许会觉得上面的例子太菜了,随便用几个 print 都能达到相同的效果,现在我们来整点花样以实现一些使用普通输出无法达到的效果。
说再多的话也不如一张图来的实际:

如果我们想要实现图中Game over!窗口,可以使用newwin方法:

除了curses.newwin新建一个独立的窗口,我们还能在任意窗口上使用subwin或者subpad方法新建子窗口,例如stdscr.subwin、stdscr.subpad、new_win.subwin、new_win.subpad等等,其使用方法与本节中创建的new_win或者stdscr没有区别,只是新建窗口使用独立的缓存区,而子窗口和父窗口共享缓存区。
如果某个窗口会在使用后删除,最好使用 newwin 方法新建独立窗口,以防止删除子窗口造成父窗口的缓存内容出现问题。
白与黑的搭配看久了也会显得单调,curses 提供了内置颜色可以让我们自定义前后背景。
在使用彩色模式之前我们需要先使用使用curses.start_corlor()进行初始化操作:
importcursesimporttimestdscr = curses.initscr()stdscr.nodelay(False)curses.noecho()# 初始化彩色模式curses.start_color()# 在1号位置添加前景色是绿色,背景色是黑色的彩色对儿curses.init_pair(1, curses.COLOR_GREEN, curses.COLOR_BLACK)# 在一行一列处显示文字,使用 1号 色彩搭配stdscr.addstr(1,1,"HelloGitHub!", curses.color_pair(1))# 阻塞等待按键然后结束程序stdscr.getch()curses.endwin()
需要注意的是,0号 位置颜色是默认黑白配色,无法修改

在此部分最后的最后,我们来说说如何给文字加一点文字效果:

前面说了这么多,现在终于到了我们的主菜。在这部分,我将一步步教给大家如何从零开始做出一个简单却又不失细节的贪吃蛇。
对于一个项目来讲,相比于尽快动手写下第一行代码不如先花点时间进行一些必要的设计,毕竟结构决定功能,一个项目没有一个良好的结构是没有前途的。
snake将贪吃蛇这个游戏分为了三大块:
- 界面:负责显示相关的所有工作
- 游戏流程控制:判断游戏输赢、游戏初始化等
- 蛇和食物:移动自身、判断是否死亡、是否被吃等
每一块都被做成了单独的对象,通过相互配合实现游戏。下面让我们来分别看看应该如何实现。
对于贪吃蛇游戏里面的蛇来讲,它可以做的事情有三种:移动,死亡(吃到自己,撞墙)和吃东西
围绕着这三个功能,我们可以首先写出一个简陋的蛇,其类图如图所示:

这个蛇可以检查自己是不是死亡,是不是吃了东西,以及更新自己的位置信息。
其中,body和last_body是列表,分别存储当前蛇身坐标和上一步蛇身坐标,默认列表第一个元素是蛇头。direction是当前行进方向,window_size是蛇可以活动的区域大小。
rest方法用于重置蛇的状态,它与__init__共同负责蛇的初始化工作:
Position 是我自定义的类,只有 x, y 两个属性,存储一个坐标点
在最开始我们可能只是模糊的感觉应该有这几个属性,但是对于其中的内容和初始化方法又不完全清楚,这是正常的。我们需要做的就是继续实现需要的功能,在实践中添加和完善最初的构想。
之后,我们从继续上到下实现,对照类图,我们接下来应该实现一下update_snake_pos即 更新蛇的位置,这部分非常简单:
defupdate_snake_pos(self)-None:# 这个函数在文章下方,获得蛇在 x, y 方向上分别增加多少dis_increment_factor =self.get_dis_inc_factor()# 需要注意,这里要用深拷贝self.last_body = copy.deepcopy(self.body)# 先移动蛇头,然后蛇身依次向前forindex, iteminenumerate(self.body):ifindex1: item.x += dis_increment_factor.x item.y += dis_increment_factor.yelse:# 剩下的部分要跟着前一部分走item.x =self.last_body[index -1].x item.y =self.last_body[index -1].y
其实 last_body 可以只记录最后一次修改的身体,这里我偷了个懒
在这里有一个细节,如果我们是第一次写这个函数,为了让蛇头能够正确的按照玩家操作移动,我们需要知道蛇头元素在 x, y 方向上各移动了多少。
最简单的方法是直接一串 if-elif,判断方向再相加:
ifself.direction==LEFT:head.x-=1elifself.direction==RIGHT:head.x+=1....
但是这样的问题在于,如果我们的需求更改直接修改这样的代码会让人很痛苦。
所以在这里更好的解决办法是使用一个dis_increment_factor存储蛇再 x 和 y 上各移动多少,并且新建一个函数get_dis_inc_factor进行判断:
defget_dis_inc_factor(self)-Position:# 初始化dis_increment_factor=Position(0,0)# 修改每个方向上的速度ifself.direction==game_config.D_Up:dis_increment_factor.y=-1elifself.direction==game_config.D_Down:dis_increment_factor.y=1elifself.direction==game_config.D_Left:dis_increment_factor.x=-1elifself.direction==game_config.D_Right:dis_increment_factor.x=1returndis_increment_factor
当然了,这么做或许有点多余,但是努力做到一个函数只做一件事情能帮助化简我们的代码,降低写出又臭又长还难调试代码的可能性。
解决了移动问题,下一步就是考虑贪吃蛇如何吃到食物了,在这里我们用check_eat_food和eat_food两个函数完成:
在这里,foods是一个存储着所有食物位置信息的列表,每次蛇体移动后都会调用check_eat_food函数检查是不是吃到了某一个食物。
可以发现,检查是不是「吃到」和「吃下去」这两个动作我分为了两个函数,以做到每个函数「一心一意」方便后期修改。
现在,我们的蛇已经能跑能吃了。但是作为一只能照顾自己的贪吃蛇,我们还需要能够判断当前自身状态,比如最基本的我需要知道我刚刚是不是咬到自己了,只需要看看蛇头是不是移动到了身体里面:
defcheck_eat_self(self)-bool:returnself.body[0]inself.body[1:]# 判断蛇头是不是和身体重合
或者我想知道是不是跑得太快而撞了墙:
defcheck_hit_wall(self)-bool:# 是不是在上下边框之间is_between_top_bottom =self.window_size.y -1self.body[0].y0# 是不是在左右边框之间is_between_left_right =self.window_size.x -1self.body[0].x0# 返回 是 或者 不是 撞了墙returnnot(is_between_top_bottomandis_between_left_right)
这些功能都是简单得不能再简单了,但是要相信自己,就是这么简单的几行代码就能实现一个听你指挥能做出复杂动作的蛇
上一节中我们实现了游戏里的第一位角色:蛇。为了将它显示出来我们现在需要将我们的命令行改造成一块「画板」。

是不是觉得有些眼花缭乱以至于感觉无从下手?其实Graphic类方法虽多但是大多数方法只是执行一个特定的功能而已,而且每次更新游戏只需要调用draw_game方法即可:
defdraw_game(self,snake:Snake, foods, lives, scores, highest_score)-None:# 清理窗口字符self.window.erase()# 绘制帮助信息self.draw_help()# 更新当前帧率self.update_fps()# 绘制帧率信息self.draw_fps()# 绘制生命、得分信息self.draw_lives_and_scores# 绘制边框self.draw_border()# 绘制食物self.draw_foods(foods)# 绘制蛇身体self.draw_snake_body(snake)# 更新界面self.window.refresh()# 更新界面self.game_area.refresh()# 延迟一段时间,以控制帧率time.sleep(self.delay_time)
遵循从上到下设计,从下到上实现的原则
可以看出draw_game实际上已经完成了Graphic的所有功能。
再往下深入,我们可以发现类似draw_foods、draw_snake_body实现基本一样,都是遍历坐标列表然后直接在相应位置上添加字符即可:
def draw_snake_body - None: for item in snake.body: self.game_area.addch(item.y, item.x, game_config.game_themes["tiles"]["snake_body"], self.C_snake)def draw_foods - None: for item in foods: self.game_area.addch(item.y, item.x, game_config.game_themes["tiles"]["food"], self.C_food)
将其分开实现也是为了保持代码干净易懂以及方便后期修改。draw_help、draw_fps、draw_lives_and_scores也是分别打印了不同文字信息,没有任何新的花样。
update_fps实现了帧率的估算以及调节等待时间稳定帧率:
defesp_fps(self)-bool:# 返回是否更新了fps# 每 fps_update_interval 帧计算一次ifself.frame_countself.fps_update_interval:self.frame_count +=1returnFalse# 计算时间花费time_span = time.time -self.last_time# 重置开始时间self.last_time = time.time()# 估算帧率self.true_fps =1.0/ (time_span /self.frame_count)# 重置计数self.frame_count =0returnTruedefupdate_fps(self)-None:# 如果重新估计了帧率ifself.esp_fps():# 计算误差err =self.true_fps -self.target_fps# 调节等待时间,稳定fpsself.delay_time +=0.00001* err
draw_message_window则实现了绘制胜利、失败的画面:
def draw_message_window(self, texts:list) - None:# 接收一个 str 列表text1 ="Press any key to continue."nrows =6+ len(texts)# 留出行与行之间的空隙ncols = max(*[len(len_tex)forlen_tex in texts], len) +20# 居中显示窗口x = (self.window.getmaxyx()[1] - ncols) /2y = (self.window.getmaxyx()[0] - nrows) /2pos = Position, int)# 新建独立窗口message_win = curses.newwin# 阻塞等待,实现任意键继续效果message_win.nodelay(False)# 绘制文字提示# 底部文字居中pos.y = nrows -2pos.x =self.get_middle) message_win.addstr(pos.y, pos.x, text1,self.C_default)# 绘制其他信息pos.y =2fortext in texts: pos.x =self.get_middle) message_win.addstr(pos.y, pos.x, text,self.C_default) pos.y +=1# 绘制边框message_win.border()# 刷新内容message_win.refresh()# 等待任意按键message_win.getch()# 恢复非阻塞模式message_win.nodelay(True)# 清空窗口message_win.clear()# 删除窗口del message_win
这样,我们就实现了游戏动画的显示!
到目前为止,我们实现了游戏内容绘制以及游戏角色实现,本节我们来学习 snake 的最后一个内容:控制。
老规矩,敲代码之前我们应该先想一想:如果要写一个control类,他应该都包含哪些方法呢?

仔细思考也不难想到:应该有一个循环,只要没输或者没赢就一直进行游戏,每轮应该更新画面、蛇移动方向等等。这就是我们的start:
def start(self) - None:# 重置游戏self.reset()# 游戏运行标志whileself.game_flag:# 绘制游戏self.graphic.draw_game(self.snake,self.foods,self.lives,self.scores,self.highest_score)# 读取按键控制ifnotself.update_control():continue# 控制游戏速度iftime.time() -self.start_time1/game_config.snake_config["speed"]:continueself.start_time = time.time()# 更新蛇self.update_snake()
只要我们写出了start对于剩下的结构也就能轻松的实现,比如读取按键控制就是最基本的比较数字是不是一样大:
def update_control(self) - bool: key =self.graphic.game_area.getch()# 不允许 180度 转弯ifkey == curses.KEY_UPandself.snake.direction != game_config.D_Down:self.snake.direction = game_config.D_Up elif key == curses.KEY_DOWNandself.snake.direction != game_config.D_Up:self.snake.direction = game_config.D_Down elif key == curses.KEY_LEFTandself.snake.direction != game_config.D_Right:self.snake.direction = game_config.D_Left elif key == curses.KEY_RIGHTandself.snake.direction != game_config.D_Left:self.snake.direction = game_config.D_Right# 判断是不是退出elif key == game_config.keys['Q']:self.game_flag =FalsereturnFalse# 判断是不是重开elif key == game_config.keys['R']:self.reset()returnFalse
更新蛇的状态时只需要判断是不是死亡、胜利、吃到东西就可:
defupdate_snake(self)-None:self.snake.update_snake_pos() index =self.snake.check_eat_food(self.foods)ifindex != -1:# 如果吃到食物# 得分 +1self.scores +=1# 如果填满了游戏区域就胜利iflen(self.snake.body) = (self.snake.window_size.x -2) * (self.snake.window_size.y -2):# 蛇身已经填满游戏区域self.win()else:# 再放置一个食物self.span_food()# 如果死了,就看看是不是游戏结束ifnotself.snake.check_alive():self.game_over()
为了让这个包能够直接使用python snake就能直接开始游戏,我们来看一下__main__.py:
importgameg = game.Game()g.start()g.quit()
当我们尝试直接运行一个包时,Python 从__main__.py中开始执行,对于我们写好的代码,只需三行即可开始游戏!

到这里如何编写一个贪吃蛇游戏就结束啦!实际上编写一个小游戏不难,对于新手来讲难点在于如何去组织程序的结构。我所实现的只是其中的一种方法,每个人对于游戏结构理解不同所写出的代码也会不同。但无论怎样,我们都应该遵循一个目标:尽量遵循代码规范,养成良好的风格。这样不仅利于别人阅读你的代码,也利于自己排查 bug、增加新的功能。