使用 Stdin 和 Stdout

作者: Dave Taylor

之前,我错误地将我的专栏命名为 “SIGALRM 定时器和 Stdin 分析”。 结果表明,当我完成写作时,我花了很多时间谈论 SIGALRM 以及如何设置定时器以避免脚本永远挂起,但我实际上从未触及 stdin 分析的主题。 哎呀。

那么这次,让我们从这个主题开始。 这里要模拟的行为是许多实用程序都在做的事情,而您却没有太注意:如果它们的输入或输出是管道或文件,它们的行为与输入或输出是 stdin(键盘)或 stdout(屏幕)时的行为不同。 尝试 lsls|cat 来看看我的意思。

test 命令在这方面有一个有用的标志:-t。 来自手册页


True if the file whose file descriptor number is
file_descriptor is open and is associated with a terminal.

值得注意的是,文件描述符 #0 是 stdin; #1 是 stdout,#2 是 stderr(分别发音为“标准输入”、“标准输出”和“标准错误”)。 这就是为什么使用 >& 通过文件描述符重定向适用于 2>&1 以使错误消息像常规输出消息一样转到 stdout 的原因。

回到主题——在实践中,-t 测试可以像这样使用


#!/bin/sh
if [ -t 0 ]; then
  echo script running interactively
else
  echo stdin coming from a pipe or file
fi

这很容易测试


$ sh inter.sh
script running interactively
$ sh inter.sh < inter.sh
stdin coming from a pipe or file
$ cat inter.sh | sh inter.sh
stdin coming from a pipe or file

完美。 现在,如何识别输出是否是交互式终端、文件或管道? 事实证明,您可以使用相同的基本测试,只需将文件 ID 0 替换为 #1


if [ -t 1 ] ; then
  echo output going to the screen
else
  echo output redirected to a file or pipe
fi

结果


$ sh inter.sh
script running interactively
output going to the screen
$ sh inter.sh | cat
script running interactively
output redirected to a file or pipe
$ sh inter.sh > output.txt
$ cat output.txt
script running interactively
output redirected to a file or pipe

非常酷,实际上。

让我们稍微回顾一下,在离开这个主题之前再看一下文件重定向。

我已经谈到了将 stderr 重定向到 stdout 的常用技巧 2>&1——这在命令行上非常有用。 您还可以将 shell 脚本中特定行的输出重定向到 stderr,这样即使 stdout 被发送到管道或文件,您的错误消息也会发送到屏幕


echo Error: this is an error message >&2

但是,如果您想让您的脚本强制 stdout 到特定目标,而不管有人在命令行上做什么,该怎么办? 这是可以做到的——当然——尽管它涉及一种非常不同的方法:使用 exec 命令。

在最基本的情况下,exec 调用就像一个子 shell 调用(这实际上是每次您调用任何系统命令(如 lsfmt)时发生的事情),但它是现有 shell 被指定的命令替换,从而有效地终止当前进程。 例如,如果您有一个 shell 脚本为外部调用设置了特定参数,您可以以以下内容结束它


exec $cmd $args

并且您在原始脚本中该点之后可能拥有的任何内容都将被抛弃,因为脚本不再运行,它被 $command 替换。

但是 exec 实际上比这更细致,特别是,它的行为的一个怪癖给出了我们寻求的解决方案:exec 将 stdin、stdout 和 stderr 的所有当前分配替换为在调用中指定的分配。

这就是解决方案,将 stdout 重定向到一个文件


exec > output.txt

在实践中,您可以通过此代码片段了解它的工作原理


echo This is stdout
exec > output.txt
echo This is still stdout but goes elsewhere

让我们实际上将一些不同的东西放在这个脚本中,这样您就可以看到这一切是如何协同工作的


echo this goes to stdout
echo and this goes to stderr >&2
exec > output.txt
echo This is still stdout but goes elsewhere
echo but where does this go\? >&2
exec date
echo this script is kaput

这是您运行程序时发生的情况


$ sh test.sh
this goes to stdout
and this goes to stderr
but where does this go?

但是,output.txt 中实际上有什么?


$ cat output.txt

这仍然是 stdout,但它去了其他地方


Sun Oct  7 10:29:56 MDT 2012

有趣。 请注意,正如预期的那样,“this script is kaput”永远不会显示出来,因为一旦 exec 调用外部程序(在本例中为 date),脚本本身就完成了,因为它的进程已被 date 程序替换。

请注意,exec 仅重定向了 stdout,因此最后的错误消息仍然会显示在屏幕上。 想要将 stdout 和 stderr 都重定向到文件吗? 这实际上只是一个字符的改变! 使用以下内容代替上面的 exec 重定向


exec &> output.txt

这很容易,不是吗?

现在,如果用户已将 stdout 重定向到一个文件,但您仍然希望它无论如何都显示在屏幕上,情况又如何呢? 这是通过 exec 调用中的另一个序列完成的:1>&2,它将 stdout 重定向到 stderr。

让我们看一下与上面相同的脚本,使用 exec 1>&2。 这是发生的事情


$ sh test2.sh > /dev/null
and this goes to stderr
This is still stdout but goes elsewhere
but where does this go?
Sun Oct  7 10:47:44 MDT 2012

非常酷,对吧?

这个月就到这里。 与往常一样,如果您有任何有趣的脚本项目、挑战或想法,请通过 https://linuxjournal.cn/contact 给 我留言,我会看一看。 始终欢迎您的输入!

此外,如果您有非凡的记忆力,您可能会记得 Mitch Frazier 在 2010 年期间在Linux Journal的 Upfront 部分撰写了关于类似主题的文章,但他的方法比我的方法复杂得多。 对不起,Mitch!

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

加载 Disqus 评论