字母和单词的随机排序

作者:Dave Taylor

你可以拖着脚步走,也可以洗牌,但是你可以随机排序字符吗?Dave 的最新专栏探讨了这种可能性。

我最近的几篇文章描述了构建一个相当复杂的密码生成器,但有一件事除外:我从未真正达到对最终结果进行加扰以增加第二层随机性的程度。我通过说这是一个留给读者练习的习题来回避这个问题,但实际上,这是一个非常有趣的问题,所以让我们在这里看看它。

你可以使用方便的 Linux 命令 rev 反转一个单词,就像这样


$ echo "hello from the other side" | rev
edis rehto eht morf olleh

你也可以反转文件中的行,以便最后一行首先显示,倒数第二行其次显示,以此类推


$ cat -n test.me | sort -rn | cut -f2-
entering along with him.
enough to prevent a swirl of gritty dust from
glass doors of Victory Mansions, though not quickly
escape the vile wind, slipped quickly through the
chin nuzzled into his breast in an effort to
clocks were striking thirteen. Winston Smith, his
It was a bright cold day in April, and the

即使是倒过来的,你也能认出这段开篇段落,对吧?“时钟敲了十三下”只能是乔治·奥威尔的警示故事《1984》。

注意:有一个名为 tac 的 Linux 命令,它提供了反向 cat,也可以完成这项工作,但我一直很喜欢 sort -rn 命令,所以我想演示如何在管道中使用它来实现相同的结果。

如果想以完全随机的顺序获取此文件的行,该怎么办?有一个命令可以做到这一点——至少在 Linux 中:shuf。但是,它在 Mac OS X 命令行中不可用,因此,如果您在家中使用 Mac 系统进行操作,那么您就遇到了障碍。对此感到抱歉。不过,我在本文末尾提供了一个替代方案,所以不要绝望!

如果您使用的是 Linux 系统(毕竟这是 Linux Journal),那么请查看一下


$ cat test.me | shuf
clocks were striking thirteen. Winston Smith, his
entering along with him.
glass doors of Victory Mansions, though not quickly
escape the vile wind, slipped quickly through the
enough to prevent a swirl of gritty dust from
chin nuzzled into his breast in an effort to
It was a bright cold day in April, and the

所以这些命令都已准备就绪,但是在一行中随机排序字母呢?这可以使用之前演示的 shuf 命令来完成,但是单个行还不太适合 shuf 处理。

你可以使用未被充分重视的 fold 命令来分解单词,就像这样


$ echo "Winston" | fold -w1
W
i
n
s
t
o
n

现在单词已被分解为按字母排列的行,shuf 命令已准备好承担任务并根据需要随机化行(字母)


$ echo "Winston" | fold -w1 | shuf
n
t
i
n
o
W
s

完美。现在,还有最后一步:将它们全部缝合在一起,使其成为一个单词,而不是一堆非常短的行。也许令人惊讶的是,tr 命令可以胜任这项任务,因为你真正需要做的就是去掉每行末尾的换行符。回想一下,tr-d 标志指示它删除指定字母的任何出现。所以,这是随机排序单词字母的完整命令


$ echo "Winston" | fold -w1 | shuf | tr -d '\n'
itoWnns$

为什么 $ 提示符会附加到随机排序的单词上?因为 tr 调用会删除所有换行符,甚至包括原本会终止最后一行的换行符。有几种方法可以解决这个问题,但最简单的方法是做一些类似脚本的事情。实际上,让我们编写一个微型脚本


$ cat scramble.sh
#!/bin/sh

# scramble - scramble whatever's specified on command line
#      usage: scramble word or phrase

echo "$*" | fold -w1 | shuf | tr -d '\n'
echo ""
exit 0

我也可以使用一些中间变量,但是正如你所看到的,在这种情况下没有必要。这真的很简单,现在你可以得到一些有趣的结果


$ sh scramble.sh Winston Smith
Shnoi tWstmin

当您使用相同的输入多次调用它时,这变得特别有趣。每次都应该不同,对吗?事实证明,确实如此


$ sh scramble.sh Winston Smith
nnih ttiWmoSs
$ sh scramble.sh Winston Smith
mnnWiosthS it
$ sh scramble.sh Winston Smith
nsmniott WhiS

这正是密码生成器所需要的,所以现在脚本终于完成了,并且可以通过添加这个简单的脚本来使用了。

而且,如果您没有 shuf

那么,对于那些没有运行 Linux 但正在使用其他 *nix 的人呢?您可以通过几种方法获取 shuf 命令或其等效命令,其中一种方法是安装整个 GNU coreutils 软件包。事实证明,Python 也可以用一行代码复制基本功能。是的,一行代码


python -c 'import sys, random; L = sys.stdin.readlines();
 ↪random.shuffle(L); print "".join(L),'

现在,我对 Python 一无所知,所以我无法解释发生了什么,但是很容易验证这确实有效


$ cat test.me | python -c 'import sys, random; L =
 ↪sys.stdin.readlines(); random.shuffle(L); print "".join(L),'
entering along with him.
clocks were striking thirteen. Winston Smith, his
escape the vile wind, slipped quickly through the
It was a bright cold day in April, and the
enough to prevent a swirl of gritty dust from
chin nuzzled into his breast in an effort to
glass doors of Victory Mansions, though not quickly

不错。感谢 superuser.com 上的 Cristian Ciupitu 提供这段代码片段,我也在这里重新发布。

而且,现在有了所有这些随机性和随机排序替代方案,您可能想深入研究随机性有多随机的问题?或者可能不想。事实证明,这是一个相当棘手的问题,也是计算机科学研究的一个丰富领域。

Dave Taylor 长期以来一直在 UNIX 和 Linux 系统上编写 shell 脚本。他是《Learning Unix for Mac OS X》和《Wicked Cool Shell Scripts》的作者。您可以在 Twitter 上找到他 @DaveTaylor,也可以通过他的技术问答网站联系他:Ask Dave Taylor

加载 Disqus 评论