如何解决如果标准输入为空,如何打破管道?
如果标准输入为空,我想打破整个管道。我尝试结合 xargs -r 和 tee,这意味着如果 stdin 为空则不打印和写入,但它失败了
...| upstream commands | xargs -r tee output.txt | downstream commands | ...
感谢任何反馈。
解决方法
实际上您无法有条件地终止 bash 管道。管道中的所有命令同时启动。然而,有一种工具可以帮助您创建条件管道。在 moreutils
中,您可以找到工具 ifne
,当且仅当输入 /dev/stdin
不为空时,它才会执行命令。所以你可以这样写:
$ command1 | ifne command2 | ifne command3 | ifne command4
此处所有命令 ifne
和 command1
都是同时启动的。只有当 ifne
通过 /dev/stdin
接收到输入时,它才会启动其各自的 commandx
如果命令失败,管道会中断。您可以在两者之间添加 grep 来实现这一点。一个例子:
$ echo ok | awk '{print $0,"1"}' | awk '{print $0,"2"}' | awk '{print $0,"3"}'
ok 1 2 3
现在添加grep:
$ echo ok | grep -Ei '^.+$' | awk '{print $0,"3"}'
ok 1 2 3
并测试空回声:
$ echo | awk '{print $0,"3"}'
1 2 3
$ echo | grep -Ei '^.+$' | awk '{print $0,"3"}'
看起来这行得通,但实际上行不通,确实很有趣,那么明显的管道不适合这里,试试这个方法:
#!/bin/bash
set -x
fun(){
data=$(echo "$1"); [[ $data ]] && data=$(awk '{print $0,1}' <<< "$data") || return 1; [[ $data ]] && data=$(awk '{print $0,2}' <<< "$data") || return 1; [[ $data ]] && data=$(awk '{print $0,3}' <<< "$data") || return 1; echo "$data"
}
fun ok
fun
测试:
$ ./test
+ fun ok
++ echo ok
+ data=ok
+ [[ -n ok ]]
++ awk '{print $0,1}'
+ data='ok 1'
+ [[ -n ok 1 ]]
++ awk '{print $0,2}'
+ data='ok 1 2'
+ [[ -n ok 1 2 ]]
++ awk '{print $0,3}'
+ data='ok 1 2 3'
+ echo 'ok 1 2 3'
ok 1 2 3
+ fun
++ echo ''
+ data=
+ [[ -n '' ]]
+ return 1
更具可读性的变体:
#!/bin/bash
set -x
fun(){
data=$(echo "$1")
[[ $data ]] && data=$(awk '{print $0,1}' <<< "$data") || return 1
[[ $data ]] && data=$(awk '{print $0,2}' <<< "$data") || return 1
[[ $data ]] && data=$(awk '{print $0,3}' <<< "$data") || return 1
echo "$data"
}
fun ok
fun
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。