一步步教你用Python实现2048小游戏


Posted in Python onJanuary 19, 2017

前言

2048游戏规则:简单的移动方向键让数字叠加,并且获得这些数字每次叠加后的得分,当出现2048这个数字时游戏胜利。同时每次移动方向键时,都会在这个4*4的方格矩阵的空白区域随机产生一个数字2或者4,如果方格被数字填满了,那么就GameOver了。

主逻辑图

一步步教你用Python实现2048小游戏

逻辑图解:黑色是逻辑层,蓝色是外部方法,红色是类内方法,稍后即可知道~

一步步教你用Python实现2048小游戏

下面容我逐行解释主逻辑main()函数,并且在其中穿叉外部定义的函数与类。

主逻辑代码解读(完整代码见文末)

主逻辑main如下,之后的是对主函数中的一些方法的解读:

def main(stdscr):
 def init():
 #重置游戏棋盘
 game_field.reset()
 return 'Game'

 def not_game(state):
 #画出 GameOver 或者 Win 的界面
 game_field.draw(stdscr)
 #读取用户输入得到action,判断是重启游戏还是结束游戏
 action = get_user_action(stdscr)
 responses = defaultdict(lambda: state) #默认是当前状态,没有行为就会一直在当前界面循环
 responses['Restart'], responses['Exit'] = 'Init', 'Exit' #对应不同的行为转换到不同的状态
 return responses[action]

 def game():
 #画出当前棋盘状态
 game_field.draw(stdscr)
 #读取用户输入得到action
 action = get_user_action(stdscr)

 if action == 'Restart':
  return 'Init'
 if action == 'Exit':
  return 'Exit'
 if game_field.move(action): # move successful
  if game_field.is_win():
  return 'Win'
  if game_field.is_gameover():
  return 'Gameover'
 return 'Game'


 state_actions = {
  'Init': init,
  'Win': lambda: not_game('Win'),
  'Gameover': lambda: not_game('Gameover'),
  'Game': game
 }

 curses.use_default_colors()
 game_field = GameField(win=32)

 state = 'Init'

 #状态机开始循环
 while state != 'Exit':
 state = state_actions[state]()

逐条解读(代码框内会标注是来自外部,无标注则是来自内部):定义主函数

def main(stdscr):
def init():
 #重置游戏棋盘
 game_field.reset()

reset出自外部定义的类,game_field=GameField的一个方法reset:

  外部:

def reset(self):
 if self.score > self.highscore:
  self.highscore = self.score
 self.score = 0
 self.field = [[0 for i in range(self.width)] for j in range(self.height)]
 self.spawn()
 self.spawn()
#其中highscore为程序初始化过程中定义的一个变量。记录你win游戏的最高分数记录。
return 'Game'

返回一个游戏进行中的状态。game_field=GameField状态在后面有定义:

主函数底部定义:

state_actions = {
  'Init': init,
  'Win': lambda: not_game('Win'),
  'Gameover': lambda: not_game('Gameover'),
  'Game': game
 }
def not_game(state):
 #画出 GameOver 或者 Win 的界面
 game_field.draw(stdscr)

draw是导入的类game_field=GameField中的方法:

#来自外部类
 def draw(self, screen):
 help_string1 = '(W)Up (S)Down (A)Left (D)Right'
 help_string2 = ' (R)Restart (Q)Exit'
 gameover_string = '  GAME OVER'
 win_string = '  YOU WIN!'
#定义各个字符串
 def cast(string):
  screen.addstr(string + '\n')

 def draw_hor_separator():
  line = '+' + ('+------' * self.width + '+')[1:]
  separator = defaultdict(lambda: line)
  if not hasattr(draw_hor_separator, "counter"):
  draw_hor_separator.counter = 0
  cast(separator[draw_hor_separator.counter])
  draw_hor_separator.counter += 1

 def draw_row(row):
  cast(''.join('|{: ^5} '.format(num) if num > 0 else '| ' for num in row) + '|')

 screen.clear()
 cast('SCORE: ' + str(self.score))
 if 0 != self.highscore:
  cast('HGHSCORE: ' + str(self.highscore))
 for row in self.field:
  draw_hor_separator()
  draw_row(row)
 draw_hor_separator()
 if self.is_win():
  cast(win_string)
 else:
  if self.is_gameover():
  cast(gameover_string)
  else:
  cast(help_string1)
 cast(help_string2)
#这里面的draw方法的字函数我就不做多的解释了,很简单的一些概念。
#但是又运用到了很优秀的精简代码。
#有的地方建议去查一下python的一些高级概念,我就不做多的介绍了。

这里面的draw方法的字函数我就不做多的解释了,很简单的一些概念。

但是又运用到了很优秀的精简代码。

有的地方建议去查一下python的一些高级概念,我就不做多的介绍了。

#读取用户输入得到action,判断是重启游戏还是结束游戏
 action = get_user_action(stdscr)

读取用户行为,函数来自于代码初始的定义

#来自外部定义的函数
def get_user_action(keyboard): 
 char = "N"
 while char not in actions_dict: 
 char = keyboard.getch()
 return actions_dict[char]

在结尾处,也即是主函数执行的第三步,定义了state = state_actions[state]()这一实例:

#主函数底部:
 state = 'Init'

 #状态机开始循环
 while state != 'Exit':
 state = state_actions[state]()
responses = defaultdict(lambda: state) #默认是当前状态,没有行为就会一直在当前界面循环
 responses['Restart'], responses['Exit'] = 'Init', 'Exit' #对应不同的行为转换到不同的状态
 return responses[action]
def game():
 #画出当前棋盘状态
 game_field.draw(stdscr)
 #读取用户输入得到action
 action = get_user_action(stdscr)

 if action == 'Restart':
  return 'Init'
 if action == 'Exit':
  return 'Exit'
 if game_field.move(action): # move successful
  if game_field.is_win():
  return 'Win'
  if game_field.is_gameover():
  return 'Gameover'
 return 'Game'
#game()函数的定义类似于上面已经讲过的not_game(),只是game()有了内部循环
#即如果不是Restart/Exit或者对move之后的状态进行判断,如果不是结束游戏,就一直在game()内部循环。

game()函数的定义类似于上面已经讲过的not_game() ,只是game()有了内部循环,即如果不是Restart/Exit或者对move之后的状态进行判断,如果不是结束游戏,就一直在game()内部循环。

state_actions = {
  'Init': init,
  'Win': lambda: not_game('Win'),
  'Gameover': lambda: not_game('Gameover'),
  'Game': game
   }

 curses.use_default_colors()
 game_field = GameField(win=32)


 state = 'Init'

 #状态机开始循环
 while state != 'Exit':
 state = state_actions[state]()
#此处的意思是:state=state_actions[state] 可以看做是:
#state=init()或者state=not_game(‘Win')或者是另外的not_game(‘Gameover')/game()

此处的意思是:state=state_actions[state] 可以看做是:state=init()或者state=not_game(‘Win')或者是另外的not_game(‘Gameover')/game()

废话不多说,上一个我的成功的图,另外,可以通过设置最后几行中的win=32来决定你最终获胜的条件!

一步步教你用Python实现2048小游戏

完整代码

#-*- coding:utf-8 -*-
import curses
from random import randrange, choice # generate and place new tile
from collections import defaultdict
letter_codes = [ord(ch) for ch in 'WASDRQwasdrq']
actions = ['Up', 'Left', 'Down', 'Right', 'Restart', 'Exit']
actions_dict = dict(zip(letter_codes, actions * 2))
def transpose(field):
 return [list(row) for row in zip(*field)]

def invert(field):
 return [row[::-1] for row in field]

class GameField(object):
 def __init__(self, height=4, width=4, win=2048):
 self.height = height
 self.width = width
 self.win_value = win
 self.score = 0
 self.highscore = 0
 self.reset()

 def reset(self):
 if self.score > self.highscore:
  self.highscore = self.score
 self.score = 0
 self.field = [[0 for i in range(self.width)] for j in range(self.height)]
 self.spawn()
 self.spawn()

 def move(self, direction):
 def move_row_left(row):
  def tighten(row): # squeese non-zero elements together
  new_row = [i for i in row if i != 0]
  new_row += [0 for i in range(len(row) - len(new_row))]
  return new_row

  def merge(row):
  pair = False
  new_row = []
  for i in range(len(row)):
   if pair:
   new_row.append(2 * row[i])
   self.score += 2 * row[i]
   pair = False
   else:
   if i + 1 < len(row) and row[i] == row[i + 1]:
    pair = True
    new_row.append(0)
   else:
    new_row.append(row[i])
  assert len(new_row) == len(row)
  return new_row
  return tighten(merge(tighten(row)))

 moves = {}
 moves['Left'] = lambda field:    \
  [move_row_left(row) for row in field]
 moves['Right'] = lambda field:    \
  invert(moves['Left'](invert(field)))
 moves['Up'] = lambda field:    \
  transpose(moves['Left'](transpose(field)))
 moves['Down'] = lambda field:    \
  transpose(moves['Right'](transpose(field)))

 if direction in moves:
  if self.move_is_possible(direction):
  self.field = moves[direction](self.field)
  self.spawn()
  return True
  else:
  return False

 def is_win(self):
 return any(any(i >= self.win_value for i in row) for row in self.field)

 def is_gameover(self):
 return not any(self.move_is_possible(move) for move in actions)

 def draw(self, screen):
 help_string1 = '(W)Up (S)Down (A)Left (D)Right'
 help_string2 = ' (R)Restart (Q)Exit'
 gameover_string = '  GAME OVER'
 win_string = '  YOU WIN!'
 def cast(string):
  screen.addstr(string + '\n')

 def draw_hor_separator():
  line = '+' + ('+------' * self.width + '+')[1:]
  separator = defaultdict(lambda: line)
  if not hasattr(draw_hor_separator, "counter"):
  draw_hor_separator.counter = 0
  cast(separator[draw_hor_separator.counter])
  draw_hor_separator.counter += 1

 def draw_row(row):
  cast(''.join('|{: ^5} '.format(num) if num > 0 else '| ' for num in row) + '|')

 screen.clear()
 cast('SCORE: ' + str(self.score))
 if 0 != self.highscore:
  cast('HGHSCORE: ' + str(self.highscore))
 for row in self.field:
  draw_hor_separator()
  draw_row(row)
 draw_hor_separator()
 if self.is_win():
  cast(win_string)
 else:
  if self.is_gameover():
  cast(gameover_string)
  else:
  cast(help_string1)
 cast(help_string2)

 def spawn(self):
 new_element = 4 if randrange(100) > 89 else 2
 (i,j) = choice([(i,j) for i in range(self.width) for j in range(self.height) if self.field[i][j] == 0])
 self.field[i][j] = new_element

 def move_is_possible(self, direction):
 def row_is_left_movable(row): 
  def change(i): # true if there'll be change in i-th tile
  if row[i] == 0 and row[i + 1] != 0: # Move
   return True
  if row[i] != 0 and row[i + 1] == row[i]: # Merge
   return True
  return False
  return any(change(i) for i in range(len(row) - 1))

 check = {}
 check['Left'] = lambda field:    \
  any(row_is_left_movable(row) for row in field)

 check['Right'] = lambda field:    \
   check['Left'](invert(field))

 check['Up'] = lambda field:    \
  check['Left'](transpose(field))

 check['Down'] = lambda field:    \
  check['Right'](transpose(field))

 if direction in check:
  return check[direction](self.field)
 else:
  return False
def main(stdscr):
 def init():
 #重置游戏棋盘
 game_field.reset()
 return 'Game'
 def not_game(state):
 #画出 GameOver 或者 Win 的界面
 game_field.draw(stdscr)
 #读取用户输入得到action,判断是重启游戏还是结束游戏
 action = get_user_action(stdscr)
 responses = defaultdict(lambda: state) #默认是当前状态,没有行为就会一直在当前界面循环
 responses['Restart'], responses['Exit'] = 'Init', 'Exit' #对应不同的行为转换到不同的状态
 return responses[action]

 def game():
 #画出当前棋盘状态
 game_field.draw(stdscr)
 #读取用户输入得到action
 action = get_user_action(stdscr)

 if action == 'Restart':
  return 'Init'
 if action == 'Exit':
  return 'Exit'
 if game_field.move(action): # move successful
  if game_field.is_win():
  return 'Win'
  if game_field.is_gameover():
  return 'Gameover'
 return 'Game'


 state_actions = {
  'Init': init,
  'Win': lambda: not_game('Win'),
  'Gameover': lambda: not_game('Gameover'),
  'Game': game
 }
 curses.use_default_colors()
 game_field = GameField(win=32)
 state = 'Init'
 #状态机开始循环
 while state != 'Exit':
 state = state_actions[state]()
curses.wrapper(main)

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流。

Python 相关文章推荐
浅析Python中的getattr(),setattr(),delattr(),hasattr()
Jun 14 Python
python爬虫入门教程--正则表达式完全指南(五)
May 25 Python
Python实现可自定义大小的截屏功能
Jan 20 Python
PyQt5打开文件对话框QFileDialog实例代码
Feb 07 Python
python写入并获取剪切板内容的实例
May 31 Python
python 用下标截取字符串的实例
Dec 25 Python
实时获取Python的print输出流方法
Jan 07 Python
Django restframework 源码分析之认证详解
Feb 22 Python
使用Python实现分别输出每个数组
Dec 06 Python
TensorFlow绘制loss/accuracy曲线的实例
Jan 21 Python
对Matlab中共轭、转置和共轭装置的区别说明
May 11 Python
selenium切换标签页解决get超时问题的完整代码
Aug 30 Python
python 开发的三种运行模式详细介绍
Jan 18 #Python
Python 3中的yield from语法详解
Jan 18 #Python
Python中的字符串操作和编码Unicode详解
Jan 18 #Python
关于Python中异常(Exception)的汇总
Jan 18 #Python
python:socket传输大文件示例
Jan 18 #Python
详解使用pymysql在python中对mysql的增删改查操作(综合)
Jan 18 #Python
python实现下载整个ftp目录的方法
Jan 17 #Python
You might like
php中++i 与 i++ 的区别
2012/08/08 PHP
PHP的Yii框架使用中的一些错误解决方法与建议
2015/08/21 PHP
PHP加密解密实例分析
2015/12/25 PHP
PHP连接MSSQL方法汇总
2016/02/05 PHP
解决laravel中日志权限莫名变成了root的问题
2019/10/17 PHP
Dom加载让图片加载完再执行的脚本代码
2008/05/15 Javascript
Javascript 网页水印(非图片水印)实现代码
2010/03/01 Javascript
clientX,pageX,offsetX,x,layerX,screenX,offsetLeft区别分析
2010/03/12 Javascript
JavaScript中json对象和string对象之间相互转化
2012/12/26 Javascript
如何使用jQuery来处理图片坏链具体实现步骤
2013/05/02 Javascript
JavaScript异步加载浅析
2014/12/28 Javascript
jQuery控制div实现随滚动条滚动效果
2016/06/07 Javascript
AngularJS 过滤器的简单实例
2016/07/27 Javascript
关于json字符串与实体之间的严格验证代码
2016/11/10 Javascript
原生js FileReader对象实现图片上传本地预览效果
2020/03/27 Javascript
Layui tree 下拉菜单树的实例代码
2019/09/21 Javascript
使用Python内置的模块与函数进行不同进制的数的转换
2016/03/12 Python
Python求均值,方差,标准差的实例
2019/06/29 Python
Python-接口开发入门解析
2019/08/01 Python
python禁用键鼠与提权代码实例
2019/08/16 Python
解决python DataFrame 打印结果不换行问题
2020/04/09 Python
Python日志logging模块功能与用法详解
2020/04/09 Python
土耳其时尚潮流在线购物网站:Trendyol
2017/10/10 全球购物
New Balance德国官方网站:购买鞋子和服装
2019/08/31 全球购物
中间件分为哪几类
2016/09/18 面试题
大学学年自我鉴定
2013/10/28 职场文书
医学院四年学习生活的自我评价
2013/11/06 职场文书
运动会领导邀请函
2014/02/05 职场文书
人力资源主管的岗位职责
2014/03/15 职场文书
孝敬父母的演讲稿
2014/05/14 职场文书
计生专干事迹
2014/05/28 职场文书
园林专业毕业生自荐信
2014/07/04 职场文书
小学数学教师研修感悟
2015/11/18 职场文书
电力企业职工培训心得体会
2016/01/11 职场文书
英镑符号 £
2022/02/17 杂记
CentOS7安装MySQL8的超级详细教程(无坑!)
2022/06/10 Servers