如何在bash中有条件地添加管道元素

ein*_*ica 1 bash pipeline pipe conditional-execution

我需要形成各种命令的管道。管道的某些元素或元素序列仅在某些条件成立时才相关。现在,我可以写:

if [[ $whatever ]]; then
   cmd1 | cmd2 | cmd3 | cmd4
else
   cmd1 | cmd4
fi
Run Code Online (Sandbox Code Playgroud)

但这意味着重复cmd1and cmd4,另外,可能有几个条件,我不想写嵌套的 if 。所以,我试着写这个:

if [[ $whatever ]]; then
    pipeline_segment="| cmd2 | cmd3"
else
    pipeline_segment=""
fi

cmd1 ${pipeline_segment} | cmd4
Run Code Online (Sandbox Code Playgroud)

但是 - 管道符号没有被解释为使用管道的指令。

我如何让 bash 也执行我想要的管道?

注意:您可以假设 bash 版本为 4 或更高,但前提是您必须这样做。

gle*_*man 6

我想你不想重复任何事情,因为它们是大而粗糙的命令。在这种情况下,使用函数来封装它们:

cmd1() { some big long command with many options and arguments; }
cmd2() { similar; }
... and so on
Run Code Online (Sandbox Code Playgroud)

你可以直接使用这个

if [[ $whatever ]]; then
   cmd1 | cmd2 | cmd3 | cmd4
else
   cmd1 | cmd4
fi
Run Code Online (Sandbox Code Playgroud)

或者,将whatever条件推入函数中:

cmd2 () { 
    if [[ whatever ]]; then
        the big long command
    else
        cat
    fi
}
cmd3() {
    if [[ whatever ]]; then
        the other big long command
    else
        cat
    fi
}
Run Code Online (Sandbox Code Playgroud)

然后,你只剩下简单的

cmd1 | cmd2 | cmd3 | cmd4
Run Code Online (Sandbox Code Playgroud)


Qua*_*odo 5

您的尝试失败,因为特殊符号在扩展后失去了它们的语法值。eval是一种方式,但正如我们所知,它很容易发生代码注入,因此最好避免。我们能试试

cmd1 |
if [ condition ]; then
   cmd2 | cmd3
else
   cat
fi | 
cmd4
Run Code Online (Sandbox Code Playgroud)

虽然我很想跳过cat,但我找不到方法。真实例子:

echo XYZABC |
if true; then
   tr Y B | tr Z C
else
   cat
fi | 
tr X A
Run Code Online (Sandbox Code Playgroud)

输出ABCABC并将条件更改为false输出AYZABC