python 基础系列篇:七、以函数方式编写一个数字华容道

文章介绍了如何使用Python编写数字华容道游戏,通过分析游戏规则,定义了包括用户输入、界面展示、移动判断等功能的函数,强调了函数在代码组织和简化工作中的作用。文章还提供了完整代码,并讨论了函数定义的规律和实用性。
摘要由CSDN通过智能技术生成

python 基础系列篇:七、以函数方式编写一个数字华容道

数字华容道

嗯,就是一个简单的益智游戏,把数字按照特定规律排列,并比矩阵少一个格,用来进行移动。

具体游戏方式就不细说了,还不了解的可以自行百度一下。

正好,老顾最近是没有什么灵感,不知道用什么举例来讲解一下怎么去划分函数,减少工作,然后就在昨天,问答有小伙伴问到了数字华容道的问题。然后老顾就决定用这个做个例子来讲解。

CSDN 文盲老顾的博客https://blog.csdn.net/supewrfei

游戏分析

老规矩,我们先分析一下需要完成的内容有哪些。既然是做数字华容道,我们先列举一下,这个游戏需要什么。

1、在一个特定长和宽的矩阵里,有 长 乘 宽 减 一 个可移动的方块
2、只有空位周边的方块可移动至空位
3、游戏开始时,方块顺序是混乱的
4、指定一个最终结果,作为胜利条件,如不指定,则以横向连续为胜利条件
5、用户可以通过上下左右(wsad)来移动方块

在游戏需求列完了,我们再列举一下,我们需要做哪些准备

1、可由用户指定游戏区域的宽和高
2、每个可移动块要标记一个记号
3、接收用户输入的移动方向
4、记录移动步数
5、生成胜利指定的最终结果,用以判定胜利条件

开始编写

相信已经看过 2048 的小伙伴,对这个感觉非常熟悉了。没错,大部分的内容,可能会与 2048 有些重合,但今天的内容,使用函数才是重点哦。

我们先规划一下,我们应该需要定义哪些方法:

1、用以定义长和宽的用户输入部分
2、用以显示游戏界面的部分
3、用来进行游戏时,接收用户输入的部分及移动
4、用来进行胜利判定的部分
5、用来进行游戏衔接的一些内容,比如是否开始新游戏,是否退出游戏等

完整代码

任何游戏都有一个进入游戏、开始游戏的指令。由于我们是在 python 开发环境里写,所以进入游戏就省略了,只写一个开始即可。在开发环境外,就用 python 指定运行文件的方式来进入游戏即可。

这次,老顾就先放出完整代码,然后再进行讲解。边写代码边写博客讲解有点费劲了。

import sys
import re
import random
import copy

# 用来呈现用户界面
def ShowBoard(data):
	# 用户界面用到的制表符
    '''─┐┌└┘├┤┬┴┼│'''
    # 根据用户定义的区域大小,来确定最大数字的长度,每个数字占用位置,以此为依据计算
    length = len(str(data['blank'])) + 1
    # 输出区域顶部边界,依据是字符占位宽度 length 和 横向数字数量 data['width']
    print(('┌' + (('─' * length) + '┬') * data['width'])[:-1] + '┐')
    for i in range(data['height']):
    	# 输出每行的数字信息
        print('│' + '│'.join([str(v).rjust(length) if v != data['blank'] else ' ' * length for v in data['board'][i * data['width']:(i + 1) * data['width']]]) + '│')
        if i < data['height'] - 1:
        	# 如果不是最后一行,输出间隔行
            print(('├' + (('─' * length) + '┼') * data['width'])[:-1] + '┤')
        else:
        	# 输出底部边界
            print(('└' + (('─' * length) + '┴') * data['width'])[:-1] + '┘')
    
# 对游戏对象进行数据填充
def InitBoard(data):
    sys.stdout.flush()
    inp = input('如果想更改区域大小,请输入两个数字,以空格分开:')
    if re.fullmatch('\s*\d+\s+\d+\s*',inp):
        data['width'],data['height'] = map(int,inp.split())
    data['blank'] = data['width'] * data['height']
    data['success'] = list(range(1, data['blank'] + 1))
    data['board'] = copy.deepcopy(data['success'])
    random.shuffle(data['board'])

# 是否新开游戏
def NewGame():
    a = ''
    while a not in 'yYnN' or len(a) < 1:
        sys.stdout.flush()
        a = input('是否开始新游戏(Y/N)?')
    return a in 'yY'

# 游戏主线程
def GetInput(data):
    ShowBoard(data)
    sys.stdout.flush()
    arrow = input('请选择方向(上w下s左a右d,退出q):').lower()
    if arrow == 'q':
        return True
    # 得到当前空位所在的位置
    blank = data['board'].index(data['blank'])
    if arrow == 'w' and blank // data['width'] < data['height'] - 1:
        data['steps'] += 1
        data['board'][blank],data['board'][blank + data['width']] = data['board'][blank + data['width']],data['board'][blank]
    if arrow == 'a' and blank % data['width'] < data['width'] - 1:
        data['steps'] += 1
        data['board'][blank],data['board'][blank + 1] = data['board'][blank + 1],data['board'][blank]
    if arrow == 's' and blank // data['width'] > 0:
        data['steps'] += 1
        data['board'][blank],data['board'][blank - data['width']] = data['board'][blank - data['width']],data['board'][blank]
    if arrow == 'd' and blank % data['width'] > 0:
        data['steps'] += 1
        data['board'][blank],data['board'][blank - 1] = data['board'][blank - 1],data['board'][blank]
    if data['board'] == data['success']:
        print('你用了{}步,取得了胜利。'.format(data['steps']))
        return True

def HuaRongDao():
    while True:
        data = {
            'width' : 4,
            'height' : 4,
            'board' : [],
            'steps' : 0
            }
        if not NewGame():
            return
        InitBoard(data)
        while True:
            if GetInput(data):
                break

if __name__ == '__main__':
    HuaRongDao()

代码解说

首先,我们在 HuaRongDao 方法里定义了一个死循环,通过死循环,来保证用户不会跳出游戏。每一次循环,表示一轮新游戏。

data = {…}

然后,在循环里定义了一个初始字典,用来存放游戏数据。每轮的数据都需要重新定义。

在初始化游戏字典后,我们询问用户是否进行新游戏,如果不进行则跳出。

data[‘width’],data[‘height’]

在初始化游戏界面的方法 InitBoard 里,我们允许用户输入两个整数,来改变游戏区域大小。毕竟是小游戏,打发时间的,可以自行加难度。而我们默认的初始难度是 4 * 4 ,算是很简单的了。

data[‘blank’] = data[‘width’] * data[‘height’]

不管用户是否改变区域大小,我们之后的内容,就是根据区域大小,来填充游戏数据了,先确定最大数字是多少,将这个数字定义为空位。

data[‘success’] = list(range(1, data[‘blank’] + 1))

然后,生成一个连续序列,表示胜利时的状态。

data[‘board’] = copy.deepcopy(data[‘success’])

再然后,用深拷贝,复制一个胜利状态的数据。

random.shuffle(data[‘board’])

最后,用随机洗牌函数,将用户需要进行操作的数据打乱。

至此,游戏初始化内容完成,可以进行游戏了。

在这里,除了最初的 data 定义,其他都放在了 InitBoard 方法里。

其实最初的定义也可以放到 InitBoard ,然后 return data,在 之前定义的循环里接收这个结果。老顾随手写的是这样,就不修改了。

因为字典是一个引用型对象,所以,我们通过传递 data 这个对象,并直接修改这个对象,是相当于在原有对象上操作的,不用担心我操作的内容会丢失。

在初始化结束后,就是正式的用户交互部分了,我们定义了一个 GetInput 的方法。

而在用户输入信息前,调用了一个 ShowBoard 方法,用来显示游戏当前界面。
在这里插入图片描述

在现实了界面后,用户才会知道自己应该怎么移动。

在输入部分,限定一下输入内容,并允许跳出游戏。即只接收 asdwq 5个字符,其他字符视为无效。

blank = data[‘board’].index(data[‘blank’])

然后就是根据空位的信息,来验证是否移动方式可行。

data[‘steps’] += 1

如果可行,则移动步数加一。在这里,老顾定义的方向也不知道是否符合大家的习惯,如果不习惯,可以将 ws 互调,ad 互调。

if data[‘board’] == data[‘success’]:

最后,用户移动完成时,验证是否胜利。

这样,一个简单的数字华容道就完成了。
在这里插入图片描述

定义方法的规律

在我们定义的这几个方法里,HuaRongDao 的使用频率是最低的,他相当于游戏的主控线程。定义这个,主要是为了方便外部执行不产生冲突。

其次,频率倒数第二低的,就是 NewGame 和 InitBoard 了,每新开一轮游戏,才调用一次,如果玩上一下午,调用次数还是不少的。

再然后,就是调用频率最高的 GetInput 了,还有同样频率的 ShowBoard。

至于为什么分成两个,一个是管输出,一个是管输入控制,分开的话,逻辑就更清晰,维护更方便罢了。

最后,老顾在 GetInput 的时候,有一些情况下具有了一个 True 的返回值,在这个代码里,这个返回值就代表了游戏结束哦,不管是胜利还是退出,对游戏逻辑来说都是一样的,只是对用户反馈信息不一样罢了。

那么,大体上的朴素逻辑就出来了,就是需要多次运行的内容,做成函数或方法,不同使用频率的,则做成不同的方法。而不同用途的,或者不同功能性的,也尽量拆分开做成不同的方法,这样后续维护,也很容易定位。

在本文中,老顾就是一个举例,具体到实际,每个人都有自己的定义方法的习惯,不用照抄老顾的习惯哦。

小结

这次,我们通过一个完整的示例代码,来了解了函数、方法的使用方式,以及朴素规律,后边我们就可以自行发挥,培养自己的代码风格和书写习惯了。

多读别人的代码,是培养代码风格和熟悉习惯的办法之一。

然后,今天引用的几个包再说明一下:

1、sys 包
主要使用 sys.stdout.flush() 避免用户输入信息提示串行,造成用户输入信息时无响应
2、re 包
用正则方式判断用户是否输入了两个整数,来确定是否需要变更区域大小。如果不用正则方式,那么用户输入信息的可能性太多,做起验证也很麻烦。
3、random 包
所有使用随机数的代码都会用到,本文主要用到 random.shuffle,对迭代对象进行打乱(洗牌)处理。
4、copy 包
在复制引用类型的数据时,应该使用深拷贝,否则你可能引用的是同一个对象,最后发现数据全乱套了。

那么,今天就到这里,大家晚安。
在这里插入图片描述

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

文盲老顾

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值