那些令人怀念的经典游戏可是提高编程能力的好素材。今天就让我们仔细探索一番,怎么用 Bash 编写一个扫雷程序。
我在编程教学方面不是专家,但当我想更好掌握某一样东西时,会试着找出让自己乐在其中的方法。比方说,当我想在 shell 编程方面更进一步时,我决定用 Bash 编写一个扫雷游戏来加以练习。
如果你是一个有经验的 Bash 程序员,希望在提高技巧的同时乐在其中,那么请跟着我编写一个你的运行在终端中的扫雷游戏。完整代码可以在这个 GitHub 存储库中找到。
做好准备
在我编写任何代码之前,我列出了该游戏所必须的几个部分:
显示雷区
在扫雷中,游戏界面是一个由 2D 数组(列和行)组成的不透明小方格。每一格下都有可能藏有地雷。玩家的任务就是找到那些不含雷的方格,并且在这一过程中,不能点到地雷。这个 Bash 版本的扫雷使用 10x10 的矩阵,实际逻辑则由一个简单的 Bash 数组来完成。
首先,我先生成了一些随机数字。这将是地雷在雷区里的位置。控制地雷的数量,在开始编写代码之前,这么做会容易一些。实现这一功能的逻辑可以更好,但我这么做,是为了让游戏实现保持简洁,并有改进空间。(我编写这个游戏纯属娱乐,但如果你能将它修改的更好,我也是很乐意的。)
下面这些变量在整个过程中是不变的,声明它们是为了随机生成数字。就像下面的 a
- g
的变量,它们会被用来计算可排除的地雷的值:
接下来,我会用列(0-9)和行(a-j)显示出游戏界面,并且使用一个 10x10 矩阵作为雷区。(M[10][10]
是一个索引从 0-99,有 100 个值的数组。) 如想了解更多关于 Bash 数组的内容,请阅读这本书那些关于 Bash 你所不了解的事: Bash 数组简介。
创建一个叫 plough
的函数,我们先将标题显示出来:两个空行、列头,和一行 -
,以示意往下是游戏界面:
然后,我初始化一个计数器变量,叫 r
,它会用来记录已显示多少横行。注意,稍后在游戏代码中,我们会用同一个变量 r
,作为我们的数组索引。 在 Bash for 循环中,用 seq
命令从 0 增加到 9。我用数字(d%
)占位,来显示行号($row
,由 seq
定义):
在我们接着往下做之前,让我们看看到现在都做了什么。我们先横着显示 [a-j]
然后再将 [0-9]
的行号显示出来,我们会用这两个范围,来确定用户排雷的确切位置。
接着,在每行中,插入列,所以是时候写一个新的 for
循环了。这一循环管理着每一列,也就是说,实际上是生成游戏界面的每一格。我添加了一些辅助函数,你能在源码中看到它的完整实现。 对每一格来说,我们需要一些让它看起来像地雷的东西,所以我们先用一个点(.
)来初始化空格。为了实现这一想法,我们用的是一个叫 is_null_field
的自定义函数。 同时,我们需要一个存储每一格具体值的数组,这儿会用到之前已定义的全局数组 room
, 并用 变量 r
作为索引。随着 r
的增加,遍历所有单元格,并随机部署地雷。
最后,为了保持游戏界面整齐好看,我会在每行用一个竖线作为结尾,并在最后结束行循环:
完整的 plough
代码如下:
我花了点时间来思考,is_null_field
的具体功能是什么。让我们来看看,它到底能做些什么。在最开始,我们需要游戏有一个固定的状态。你可以随便选择个初始值,可以是一个数字或者任意字符。我最后决定,所有单元格的初始值为一个点(.
),因为我觉得,这样会让游戏界面更好看。下面就是这一函数的完整代码:
现在,我已经初始化了所有的格子,现在只要用一个很简单的函数就能得出当前游戏中还有多少单元格可以操作:
这是显示出来的游戏界面,[a-j]
为列,[0-9]
为行。
Minefield
创建玩家逻辑
玩家操作背后的逻辑在于,先从 stdin 中读取数据作为坐标,然后再找出对应位置实际包含的值。这里用到了 Bash 的参数扩展,来设法得到行列数。然后将代表列数的字母传给分支语句,从而得到其对应的列数。为了更好地理解这一过程,可以看看下面这段代码中,变量 o
所对应的值。 举个例子,玩家输入了 c3
,这时 Bash 将其分成两个字符:c
和 3
。为了简单起见,我跳过了如何处理无效输入的部分。
下面的代码会计算用户所选单元格实际对应的数字,然后将结果储存在变量中。
这里也用到了很多的 shuf
命令,shuf
是一个专门用来生成随机序列的 Linux 命令。-i
选项后面需要提供需要打乱的数或者范围,-n
选项则规定输出结果最多需要返回几个值。Bash 中,可以在两个圆括号内进行数学计算,这里我们会多次用到。
还是沿用之前的例子,玩家输入了 c3
。 接着,它被转化成了 ro=3
和 o=3
。 之后,通过上面的分支语句代码, 将 c
转化为对应的整数,带进公式,以得到最终结果 i
的值。
仔细观察这个计算过程,看看最终结果 i
是如何计算出来的:
最后结果是 33。在我们的游戏界面显示出来,玩家输入坐标指向了第 33 个单元格,也就是在第 3 行(从 0 开始,否则这里变成 4),第 3 列。
创建判断单元格是否可选的逻辑
为了找到地雷,在将坐标转化,并找到实际位置之后,程序会检查这一单元格是否可选。如不可选,程序会显示一条警告信息,并要求玩家重新输入坐标。
在这段代码中,单元格是否可选,是由数组里对应的值是否为点(.
)决定的。如果可选,则重置单元格对应的值,并更新分数。反之,因为其对应值不为点,则设置变量 not_allowed
。为简单起见,游戏中警告消息这部分源码,我会留给读者们自己去探索。
Extracting mines
如输入坐标有效,且对应位置为地雷,如下图所示。玩家输入 h6
,游戏界面会出现一些随机生成的值。在发现地雷后,这些值会被加入用户得分。
Extracting mines
还记得我们开头定义的变量,a
- g
吗,我会用它们来确定随机生成地雷的具体值。所以,根据玩家输入坐标,程序会根据(m
)中随机生成的数,来生成周围其他单元格的值(如上图所示)。之后将所有值和初始输入坐标相加,最后结果放在 i
(计算结果如上)中。
请注意下面代码中的 X
,它是我们唯一的游戏结束标志。我们将它添加到随机列表中。在 shuf
命令的魔力下,X
可以在任意情况下出现,但如果你足够幸运的话,也可能一直不会出现。
我想要游戏界面中,所有随机显示出来的单元格,都靠近玩家选择的单元格。
Extracting mines
记录已选择和可用单元格的个数
这个程序需要记录游戏界面中哪些单元格是可选择的。否则,程序会一直让用户输入数据,即使所有单元格都被选中过。为了实现这一功能,我创建了一个叫 free_fields
的变量,初始值为 0
。用一个 for
循环,记录下游戏界面中可选择单元格的数量。 如果单元格所对应的值为点(.
),则 free_fields
加一。
等下,如果 free_fields=0
呢? 这意味着,玩家已选择过所有单元格。如果想更好理解这一部分,可以看看这里的源代码。
创建游戏结束逻辑
对于游戏结束这种情况,我们这里使用了一些很巧妙的技巧,将结果在屏幕中央显示出来。我把这部分留给读者朋友们自己去探索。
最后,我们显示出玩家最关心的两行。
Minecraft Gameover
文章到这里就结束了,朋友们!如果你想了解更多,具体可以查看我的 GitHub 存储库,那儿有这个扫雷游戏的源代码,并且你还能找到更多用 Bash 编写的游戏。 我希望,这篇文章能激起你学习 Bash 的兴趣,并乐在其中。
【责任编辑:庞桂玉 TEL:(010)68476606】