如何将命令输出作为参数执行而不使用 $() 保存它?

如何将命令输出作为参数执行而不使用 $() 保存它?

这实际上是为了brew search

但为了简单起见,假设我这样做:

echo -e 'somedir\n\otherdir'
somedir
otherdir

我可以otherdir在不这样做的情况下用作参数吗?

ls $(echo -e 'somedir\n\otherdir |tail -n1)
otherdir

更新以明确目标

为了更清楚somedirotherdir转到标准输出,我可以仅在otherdir

不捕获 中的输出 $(),然后对其执行命令?

我只是想知道这是否可能,尽管我认为这是不可能的。

答案1

您始终可以使用xargs适用于此类事情的:将输出流转换为参数列表:

printf 'somedir\n\otherdir\n' |
  tail -n1 |
  xargs -rd '\n' ls -l --

请注意-d '\n',这里需要参数,因为行需要 GNU 实现xargs

如果最后一行不包含空格、引号或反斜杠,则可以省略它。

-rls,如果输入为空则需要避免运行也来自 GNU xargs,但现在得到广泛支持(并将出现在 POSIX 标准的下一版本中)。


如果您不想使用命令替换,则在 中bash,您始终可以将命令输出的行读入数组并在该数组的最后一个元素上运行该命令:

readarray -t lines < <(printf 'somedir\notherdir\n')
if (( ${#lines[@]} > 0 )); then
  ls -l -- "${line[@]: -1}"
fi

或者使用perl

printf 'somedir\notherdir\n' |
  perl -lne 'exec qw(ls -l --), $_ if eof && $.'

请注意,使用命令替换时,由于它会删除所有尾随换行符,因此您无法直接区分空输入和最后一行为空。在这两种情况下,您都会得到 和 空扩展。

使用命令替换的优点之一是,您可以在决定使用其输出之前检查命令是否已成功(基于其退出状态):

die() { [ "$#" -eq 0 ] || printf>&2 '%s\n' "$@"; exit 1; }

last_line=$(
  set -o pipefail
  cmd | tail -n 1
) || die "cmd failed"

ls -l -- "$last_line"

或者您grep也可以使用它来检查输出是否非空:

if
  last_line=$(
    set -o pipefail
    cmd |
      tail -n 1 |
      grep '^'
  )
then
  ls -l -- "$last_line"
else
  echo>&2 "Either cmd failed or it didn't produce any output"
fi

grep '^'匹配行的开头。因此,如果输出至少包含一行(无论是否为空),它将返回 true。但请注意,如果输入不以换行符结尾,则根据行的 POSIX 定义,它不会被视为行,并且某些grep实现在这种情况下可能会失败,甚至会丢弃该非分隔行。

答案2

语法$()没有节省任何事物。它基本上意味着“运行此命令并使用输出作为参数”。

您的用法是正确的规范做事方式。

在较旧的 shell 中(它仍然有效),反引号(`)的作用类似

但这$()是现代的方式,并且还解决了旧方式存在的许多毛茸茸的引用问题。

这几乎可以在任何地方使用;例如

echo "The current time is: $(date)"
The current time is: Tue Nov 21 22:57:41 EST 2023

相关内容